匿名委托的叫法并不准确,准确的应该叫做匿名方法,(总之两者是一个意思啦)。前面 委托类型 中我已经提到过,委托是用于引用与其具有相同标签的方法。换句话说,您可以使用委托对象调用可由委托引用的方法(参数是方法名)。而匿名方法则是将代码块作为委托参数(参数是实现功能的代码)通过使用匿名方法,由于您不必创建单独的方法,因此减少了实例化委托所需的编码系统开销。
匿名方法是直接挂载在委托内的代码块,还是得通过使用 delegate 关键字创建委托实例来声明。
delegate void MyDelegate(int i); //声明一个委托
MyDelegate my = delegate(int i){ /* 代码块*/ }; //通过创建一个委托实例来实现一个匿名方法
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Test1
{
class Program
{ //声明一个委托
delegate void MyDelegate(string str);
//定义一个实名方法
public static void fun(string str)
{
Console.WriteLine("这是一个 {0} 方法", str);
}
static void Main(string[] args)
{
//创建一个委托实例,里面包含一个匿名方法
MyDelegate examp1 = delegate(string name)
{
Console.WriteLine("这是一个 {0} 方法",name); //代码块
};
examp1("匿名"); //调用匿名方法
MyDelegate examp2 = new MyDelegate(fun); //在委托中实名注册一个fun命名方法
examp2("实名"); //调用命名方法
}
}
}
结果:
匿名方法的参数的范围是“匿名方法块”。
如果目标在块外部,那么,在匿名方法块内使用跳转语句(如 goto、break 或 continue)是错误的。如果目标在块内部,在匿名方法块外部使用跳转语句(如goto、break 或continue)也是错误的。
以前我们使用delegate委托时必须的提前声明一个delegate类,然后向委托中注册方法,比如:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Test1
{
class Program
{
delegate string MyDelegate(string s);//声明委托
public static string Toup(string str) //定义方法
{
return str.ToUpper();
}
static void Main(string[] args)
{
string str = "abc";
MyDelegate my = Toup; //注册方法
Console.WriteLine(my(str));//调用方法 结果 ABC
}
}
}
如果当条件不允许我们在程序中声明delegate类,但又需要使用委托时,我们该怎么办呢? 此时我们可以考虑使用Func委托。Func
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Test1
{
class Program
{
public static string Toup(string str) //定义方法
{
return str.ToUpper();
}
static void Main(string[] args)
{
string str = "abc";
Func change = Toup; //泛型委托
Console.WriteLine(change(str));//调用方法 结果 ABC
}
}
}
对比下,两者结果一样,但Func却比Delegate简洁了很多,但是Delegate能够加载匿名方法,比如上面的例子我们使用匿名方法:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Test1
{
class Program
{
delegate string MyDelegate(string s);//声明委托
static void Main(string[] args)
{
string str = "abc";
//创建匿名方法
MyDelegate my = delegate(string s) { return s.ToUpper(); };
Console.WriteLine(my(str)); //结果 ABC
}
}
}
Func也行吗? Func也是可以创建匿名方法的,同样的也不需要声明,如下:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Test1
{
class Program
{
static void Main(string[] args)
{
string str = "abc";
//创建匿名方法
Func change = delegate(string s) { return s.ToUpper(); };
Console.WriteLine(change(str)); //结果 ABC
}
}
}
与上面的匿名方法对比我们发现,在创建匿名方法的时候两者都是通过 delegate 来实现的(还是用到了delegate),可不可以不用delegate ,不用它是可以的,这时我们需要学习lambda 表达式
Lambda 表达式是一种可用于创建委托或表达式目录树类型的匿名方法。 通过使用 lambda 表达式,可以写入可作为参数传递或作为函数调用值返回的本地函数。若要创建 Lambda 表达式,需要在 Lambda 运算符 => 左侧指定输入参数(如果有),然后在另一侧输入表达式或语句块。 例如,lambda 表达式 x => x * x 指定名为 x 的参数并返回 x 的平方值。
所以上面的例子我们使用lambda 表达式创建匿名方法改为:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Test1
{
class Program
{
static void Main(string[] args)
{
string str = "abc";
//lambda 表达式
Func change = s => s.ToUpper(); //传入string 类型s 返回s.ToUpper();
Console.WriteLine(change(str)); //结果 ABC
}
}
}
在delegate委托类型中,我们也可以使用lambda表达式创建匿名方法:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Test1
{
class Program
{
delegate string MyDelegate(string s);//声明委托
static void Main(string[] args)
{
string str = "abc";
//lambda 表达式
MyDelegate my = s => s.ToUpper(); //传入string 类型s 返回s.ToUpper();
Console.WriteLine(my(str)); //结果 ABC
}
}
}
感谢您的阅读,欢迎您的建议与评论^_^