在WinForm中,不允许非UI线程访问UI,如果非UI线程需要跨线程调用UI控件,通常的解决办法是使用Control类中的Invoke方法,传递给该方法一个委托和委托调用的参数列表(params []object args),任何委托类型都可以,通过委托来访问UI。其内部是,非UI线程把委托送到UI线程中,让UI线程去调用这个委托。

一般一个方法的参数是委托类型,如果使用委托实例,那就还需要额外定义一个和委托签名一致的方法,比较麻烦。而我一般是使用匿名方法,因为匿名方法方便,不需要额外的声明的一个委托类型了,让代码更加简洁。但是这一次使用匿名方法却出现了错误,在编译的时候就没有被通过,我代码如下(code1):

 this.Invoke(delegate(string test) {this.Text=test;},"myCall");

编译时错误如下:

无法将 匿名方法 转换为类型“System.Delegate”,因为它不是委托类型

经过仔细思考,发现这样的写法通过编译确实是对的。首先得明白所有的委托类型都是继承System.Delegate,而Invoke方法委托类型就是Delegate,也就是说,任意的委托类型都可以作为该方法的参数。如果我们使用匿名方法,那么CLR就不知道把该匿名方法转为哪种类型的的委托,因此直接传入一个匿名函数是通不过编译的。知道了为什么错误,代码改成下面这样(code2),就能够正确的运行罗。

Action<string> updateUI = (text) => this.Text = text;
this.Invoke(updateUI,"myCall");
// 或者这样,和上面一样,还是传入Action<string>委托实例
this.Invoke(new Action<string>((text) => this.Text = text), "myCall");

在使用委托类型的时候,建议最好采用系统内部定义Action和Func两种委托类型,而且这两种类型还定义了许多个泛型版本,足够满足我们平常的应用啦。

那么在哪种情况下可以直接传入匿名方法作为参数,举个例子,在使用Linq的扩展Where方法的时候,该方法需要传入一个委托,委托的定义为Func<T,bool>,这种情况我们可以直接使用一个匿名方法,代码如下(code3):

list.Where((s) => s.StartsWith("n"));

因此可以知道,当方法的参数是具体的委托类型时,可以使用匿名方法,因为编译器知道应该把该匿名方法转为何种委托类型;而 当方法的参数为委托的基类System.Delegate时,那就需要自己实例化一个具体委托类型实例(委托类型使用.NET内部定义的,如code2),然后传入进去,这样CLR知道就是何种委托类型。

在C#3.0中有了一个叫Lambda的表达式,和匿名方法是一样效果,但是Lambda能够使代码更加紧凑,更加优异,主要是比较容易理解表达式,可读性好,关于Lambda的用法在这里就不做介绍了。上面的code2和code3的匿名方法都是采用的Lambda表达式,是不是看起来更加简洁。

无法将匿名方法转换为System.Delegate的更多相关文章

  1. 无法将 匿名方法 转换为类型“System.Delegate”,因为它不是委托类型:解决方法

    http://blog.csdn.net/xiaochongchong1248/archive/2009/11/20/4841193.aspx?1271573283 编程环境要求:VS2008/FX2 ...

  2. 关于委托:异常{ 无法将 匿名方法 转换为类型“System.Delegate”,因为它不是委托类型 }

    转自:http://www.cnblogs.com/xiaofei59/archive/2010/11/25/1887285.html 异常{ 无法将 匿名方法 转换为类型“System.Delega ...

  3. 关于委托:异常{ 无法将 匿名方法 转换为类型“System.Delegate”,因为它不是委托类型 }

    异常{ 无法将 匿名方法 转换为类型"System.Delegate",因为它不是委托类型 } 委托实际上是把方法名作为参数,但是若有好多个方法时,就要指明是哪个参数  查看如下代 ...

  4. 委托delegate 泛型委托action<> 返回值泛型委托Func<> 匿名方法 lambda表达式 的理解

    1.使用简单委托 namespace 简单委托 { class Program { //委托方法签名 delegate void MyBookDel(int a); //定义委托 static MyB ...

  5. C# 从CIL代码了解委托,匿名方法,Lambda 表达式和闭包本质

    前言 C# 3.0 引入了 Lambda 表达式,程序员们很快就开始习惯并爱上这种简洁并极具表达力的函数式编程特性. 本着知其然,还要知其所以然的学习态度,笔者不禁想到了几个问题. (1)匿名函数(匿 ...

  6. 进阶系列(8)——匿名方法与lambda表达式

    一 匿名方法的介绍     匿名方法是为了简化委托的实现,方便调用委托方法而出现的,同时,匿名方法也是学好lambda表达式的基础.在委托调用的方法中,如果方法只被调用一次,这个时候我们就没有必要创建 ...

  7. 匿名方法、Lambda表达和自定义泛型委托以及Func、Action系统泛型委托

    1.匿名方法的概念:一个方法没有具体的名称,而只有关键字delegate.方法参数.方法体.这种方法是匿名方法. 匿名方法的好处:将具体方法和委托直接关联在一起,如果我们基于委托只需要一个方法的时候, ...

  8. C#委托总结-匿名方法&Lambda表达式

    1,匿名方法 匿名方法可以在声明委托变量时初始化表达式,语法如下 之前写过这么一段代码: delegate void MyDel(string value); class Program { void ...

  9. c#委托、泛型委托和匿名方法

    题外话:别指望看第一遍书就能记住和掌握什么——请看第二遍.第三遍. 本人女猿一枚,2年工作经验,喜欢钻研,喜欢创新,闲暇之余喜欢写写博客,深知自身能力薄弱,如表达错误.不当之处请园友们多多指出,互相交 ...

随机推荐

  1. ASP.NET高并发解决方案

    一.页面静态化: 二.运用WCF,WEB SERVICE等技术实现分布式业务逻辑处理及数据访问的能力: 三.页面缓存: 四.数据缓存: 五.建立图片文件服务器:

  2. 重构第9天:提取接口(Extract Interface)

    理解:提取接口的意思是,多于一个类共同使用某个类中的方法或属性,那么我们可以把这些方法和属性提出来,作为一个单独的接口.这样的好处是解除代码间的依赖,降低耦合性. 详解: 先看重构前的代码: publ ...

  3. jquery.ajax 跨域请求webapi,设置headers

    解决跨域调用服务并设置headers 主要的解决方法需要通过服务器端设置响应头.正确响应options请求,正确设置 JavaScript端需要设置的headers信息 方能实现. 1.第一步 服务端 ...

  4. Python基础:新式类的属性访问

    一.概述 二.准备工作 1.讨论对象 2.名词解释 三.实例绑定的属性访问 1.获取属性 一般规则 参考源码 示例验证 2.设置属性 一般规则 参考源码 示例验证 3.删除属性 一般规则 参考源码 示 ...

  5. js获取url传递的参数

    获取URL带参数的JAVASCRIPT客户端解决方案 一.正则分析法.(我较喜欢使用正则)function GetQueryString(name) {var reg = new RegExp(“(^ ...

  6. 泛函编程(9)-异常处理-Option

    Option是一种新的数据类型.形象的来描述:Option就是一种特殊的List,都是把数据放在一个管子里:然后在管子内部对数据进行各种操作.所以Option的数据操作与List很相似.不同的是Opt ...

  7. python,python3

    当下主流的Linux发行版都集成了python2,直接在终端输入$python就可进入交互式解释器,或者输入$python3进入python3的解释器,当然,也可以使用这个解释器直接翻译一个pytho ...

  8. 类库LinqToExcel的介绍

            LinqToExcel是一个.net framework平台下开源项目,它主要实现了LINQ的语法查询Excel电子表格.类型之前的LINQToXXX如果你是LINQ语法糖爱好者那最适 ...

  9. 【GOF23设计模式】代理模式

    来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_代理模式.静态代理 package com.test.proxy.staticProxy; public interfac ...

  10. nodeJS中npm常见的命令

    常用的nodeJS中npm的命令:npm主要是node包管理和发布的工具.npm常用的命令:1:npm install <name> //(下载包) 下载后的包放在当前路径下面 npm i ...