欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

C#基础之匿名方法实例教程

程序员文章站 2023-12-20 17:44:16
本文以实例形式讲解了c#的匿名方法的用法,分享给大家供大家参考之用。具体如下: 匿名方法是c# 2.0的语言新特性。首先看个最简单的例子: class prog...

本文以实例形式讲解了c#的匿名方法的用法,分享给大家供大家参考之用。具体如下:

匿名方法是c# 2.0的语言新特性。首先看个最简单的例子:

class program 
{ 
 static void main(string[] args) 
 { 
  list<string> names = new list<string>(); 
  names.add("sunny chen"); 
  names.add("kitty wang"); 
  names.add("sunny crystal"); 
   
  list<string> found = names.findall( new predicate<string>(namematches)); 
 
  if (found != null) 
  { 
   foreach (string str in found) 
    console.writeline(str); 
  } 
 } 
 
 static bool namematches(string name) 
 { 
  return name.startswith("sunny", stringcomparison.ordinalignorecase); 
 }
}

这段代码在开始的时候初始化了一个字符串列表(string list),然后通过列表的findall方法来查找以“sunny”起始的字符串,最后将所查找到的所有结果输出。

我们需要着重介绍list<t>的findall方法。这个方法是一个参数为predicate<t>类型、返回值为list<t>类型的函数。注意,predicate<t>是一个泛型委托,它指代这样一些函数:这些函数仅有一个t类型的参数,并且返回值是布尔类型。通过reflector等工具,我们可以看到predicate<t>的定义如下:

public delegate bool predicate<t>(t obj); 

至此我们也多少能够猜到findall方法的具体实现。即针对list<t>中的每个元素,调用predicate<t>所指代的函数,如果函数返回为true,则将其加入新建的列表中。遍历完所有的元素后,将新建的列表返回给调用者。如下:

public list<t> findall<t>(predicate<t> match) 
{ 
 list<t> ret = new list<t>(); 
 foreach (t elem in items) 
 { 
  if (match(elem)) 
   ret.add(elem); 
 } 
 return ret; 
} 

因此,针对上面的例子,要调用findall方法,我们必须先定义一个参数为string类型,返回值为布尔类型的函数,在这个函数中,对参数进行条件判断,如果符合条件(也就是以“sunny”作为起始字符串),那么就返回true,否则返回false。最后再将这个函数作为参数传递给findall。于是也就得到了最上面的代码。

在上面的例子中,为了调用findall方法,我们不得不新定义一个函数,其实这个函数除了findall方法要用外,别的地方都几乎很少使用到它,你还不得不给它起个名字。如果程序中有多处需要调用findall方法,或者类似的情况,那么整个程序也就会出现一大批“只有一个地方使用”的函数,使得代码难于阅读和维护。

由于存在这样的问题,c# 2.0引入了匿名方法。开发人员在实现方法的时候,只需要给出方法的参数列表(甚至也可以不给)以及方法具体实现,而不需要关心方法的返回值,更不必给方法起名字。最关键的是,只在需要的地方定义匿名方法,保证了代码的简洁。

匿名方法只在需要的地方定义,定义的时候,使用delegate关键字,后接参数列表,然后跟上用一对花括号包括起来的函数体即可。上面的代码可以重构成下面的形式:

class program 
{ 
 static void main(string[] args) 
 { 
  list<string> names = new list<string>(); 
  names.add("sunny chen"); 
  names.add("kitty wang"); 
  names.add("sunny crystal"); 
 
  list<string> found = names.findall( 
   delegate(string name) 
   { 
    return name.startswith("sunny", stringcomparison.ordinalignorecase); 
   }); 
 
  if (found != null) 
  { 
   foreach (string str in found) 
    console.writeline(str); 
  } 
 } 
 
 //static bool namematches(string name) 
 //{ 
 // return name.startswith("sunny", 
 //  stringcomparison.ordinalignorecase); 
 //} 
} 

此时,我们完全不需要namematches方法了,直接将匿名方法作为参数传递给findall方法。其实匿名方法本身还是有名字的,只是我们并不关心它究竟该取什么名字,因而.net帮我们随便取了个名字罢了。

匿名方法在c#中应用十分广泛,因为委托作为函数参数是件非常平常的事情。在定义简单的事件处理过程时,我们同样可以使用匿名方法。比如:

servicehost host = new servicehost(typeof(filetransferimpl)); 
host.opened += delegate(object sender, eventargs e) 
{ 
 console.writeline("service opened."); 
};

匿名方法可以很方便地使用本地变量,这与单独定义的命名方法相比,能够简化编程。比如上文的例子中,假如main函数里面定义了一个整型本地变量(局部变量)number,那么可以在delegate (string name)这一匿名方法定义中使用number变量。

上文提到,在定义匿名方法的时候,连参数列表都可以省略。因为编译器可以根据委托的签名来确定函数的签名,然后只要再给函数起个名字就可以了。下面的代码演示了这种使用方式:

delegate void intdelegate(int x); 
 
// 带参数的定义方式 
intdelegate d2 = delegate(int p) { console.writeline(p); }; 
// 不带参数的定义方式(当然也没带返回值) 
intdelegate d3 = delegate { console.writeline("hello."); }; 

在使用不带参数和返回值的匿名方法定义时,需要注意以下两点:

1.如果在你的匿名方法中需要对参数进行处理,那么你不能使用不定义参数列表的声明方式。也就是在定义匿名方法的时候,需要给出参数列表。
2.不带参数和返回值的匿名方法,可以被具有任何形式签名的委托所指代。

上述第一点显而易见,因为你没有定义参数列表,也就没有办法使用参数;要说明第二点,我们可以看下面的代码:

class program 
{ 
 delegate void intdelegate(int x); 
 delegate void stringdelegate(string y); 
 
 static void output(intdelegate id) 
 { 
 } 
 
 static void output(stringdelegate sd) 
 { 
 } 
 
 static void main(string[] args) 
 { 
  /* 
   * error: the call is ambiguous between 
   *  output(intdelegate) 
   *    and 
   *  output(stringdelegate) 
   */ 
  output(delegate { }); 
 } 
} 

上面的代码没法编译通过,因为编译器不知道应该将delegate { }这一匿名方法还原为由intdelegate指代的函数,还是还原为由stringdelegate指代的函数。此时只能显式给定参数列表,以便让编译器知道,我们究竟是想调用哪个output函数。

希望本文所述对大家的c#程序设计有所帮助

上一篇:

下一篇: