• 匿名方法的使用


    匿名方法相信很多人都听过,它是C#2.0的一个新特性,顾名思义,匿名方法就是没有名称的方法。那么在C#中的匿名方法有哪些好处,在C#中如何使用呢?
    匿名方法最明显的好处就是可以降低另写一个方法的工作量,另外一个好处就是可以访问调用者的变量,降低传参数的复杂度,下面就通过一些使用例子来具体看看。
    1、在事件中使用匿名方法
    下面是一个定时器的小例子,我们常规的写法如下:
    常规写法
    对于事件的处理我们需要单独写一个方法timersTimer_Elapsed,那么如果使用匿名方法,就可以省掉这个方法的定义,如下所示:
        class EventTest
        
    {
            
    public void Test()
            
    {
                System.Timers.Timer timersTimer 
    = new System.Timers.Timer();

                timersTimer.Enabled 
    = true;
                timersTimer.Interval 
    = 5000;
                timersTimer.Elapsed 
    +=
                    
    delegate(object sender, System.Timers.ElapsedEventArgs e)
                    
    {
                        Console.WriteLine(System.DateTime.Now);
                    }
    ;
                Console.ReadLine();
            }

        }

    也就是把方法的实现直接写在内部。

    2、在我们自己的代理中使用匿名方法
    比如如下一个字符串转化的例子,按照传统的方法来写的话,对于每一种转化规则都需要单独定义一个方法:
    常规写法
    我们的例子中有三种规则,那么就要定义三个方法,如果使用匿名方法的话,代码就会很简单:
        class DelegateTest
        
    {
            
    delegate string Convert(string text);

            
    public void Test()
            
    {
                
    string strText = "123451234512345";

                Test2(
    delegate(string strText2)
                    
    {
                        
    return strText2.Replace("1""A");
                    }
    , strText);
                Test2(
    delegate(string strText2)
                    
    {
                        
    return strText2.Replace("1""B");
                    }
    , strText);
                Test2(
    delegate(string strText2)
                    
    {
                        
    return strText2.Replace("1""C");
                    }
    , strText);

                Console.ReadLine();
            }


            
    private void Test2(Convert convert, string strText)
            
    {
                Console.WriteLine(convert(strText));
            }

        }

    3、优化例子2,不再传递参数
    在例子2中我们是把参数传到你们方法内部的,其实在匿名方法内部可以直接取得当前调用者的变量,节省了传递参数的代码量:
        class AnonTest
        
    {
            
    delegate string Convert();

            
    public void Test()
            
    {
                
    string strText = "123451234512345";

                Test2(
    delegate()
                    
    {
                        
    return strText.Replace("1""A");
                    }
    );
                Test2(
    delegate()
                    
    {
                        
    return strText.Replace("1""B");
                    }
    );
                Test2(
    delegate()
                    
    {
                        
    return strText.Replace("1""C");
                    }
    );

                Console.ReadLine();
            }


            
    private void Test2(Convert convert)
            
    {
                Console.WriteLine(convert());
            }

        }
    这样一来,代码是不是看起来很整洁:)

    当然代码虽然看起来很少,实际上编译器在编译时还是会生成其他方法的。也就是说匿名方法可以减少代码量,节省开发时间,但是对于性能方法没有什么提升的。
    我们通过IL DASM工具可以查看一下AnonTest类编译后的代码,会发现增加一个新类,在这个类里面生成了三个方法和一个同名的strText变量:


    然后在Test方法中会调用这些新生成的方法,Test如下所示:
    .method public hidebysig instance void  Test() cil managed
    {
      
    // 代码大小       83 (0x53)
      .maxstack  4
      .locals init ([
    0class AnonMethod.AnonTest/'<>c__DisplayClass3' '<>8__locals4')
      IL_0000:  newobj     instance 
    void AnonMethod.AnonTest/'<>c__DisplayClass3'::.ctor()
      IL_0005:  stloc.
    0
      IL_0006:  nop
      IL_0007:  ldloc.
    0
      IL_0008:  ldstr      
    "123451234512345"
      IL_000d:  stfld      
    string AnonMethod.AnonTest/'<>c__DisplayClass3'::strText
      IL_0012:  ldarg.
    0
      IL_0013:  ldloc.
    0
      IL_0014:  ldftn      instance 
    string AnonMethod.AnonTest/'<>c__DisplayClass3'::'<Test>b__0'()
      IL_001a:  newobj     instance 
    void AnonMethod.AnonTest/Convert::.ctor(object,
                                                                            native 
    int)
      IL_001f:  call       instance 
    void AnonMethod.AnonTest::Test2(class AnonMethod.AnonTest/Convert)
      IL_0024:  nop
      IL_0025:  ldarg.
    0
      IL_0026:  ldloc.
    0
      IL_0027:  ldftn      instance 
    string AnonMethod.AnonTest/'<>c__DisplayClass3'::'<Test>b__1'()
      IL_002d:  newobj     instance 
    void AnonMethod.AnonTest/Convert::.ctor(object,
                                                                            native 
    int)
      IL_0032:  call       instance 
    void AnonMethod.AnonTest::Test2(class AnonMethod.AnonTest/Convert)
      IL_0037:  nop
      IL_0038:  ldarg.
    0
      IL_0039:  ldloc.
    0
      IL_003a:  ldftn      instance 
    string AnonMethod.AnonTest/'<>c__DisplayClass3'::'<Test>b__2'()
      IL_0040:  newobj     instance 
    void AnonMethod.AnonTest/Convert::.ctor(object,
                                                                            native 
    int)
      IL_0045:  call       instance 
    void AnonMethod.AnonTest::Test2(class AnonMethod.AnonTest/Convert)
      IL_004a:  nop
      IL_004b:  call       
    string [mscorlib]System.Console::ReadLine()
      IL_0050:  pop
      IL_0051:  nop
      IL_0052:  ret
    }
     // end of method AnonTest::Test

  • 相关阅读:
    asp.net FckEditor配置
    您请求的报表需要更多信息...
    水晶报表中如何动态增加字段
    使用JavaMail发送SMTP认证的邮件给多个收信人
    vim中删除每行行尾的空格
    转载:STUN在SIP中的工作原理及过程
    转载 URL和URI的区别
    转载 Android深入浅出Binder机制
    链接静态库的时候,命令行中库和源文件的位置问题
    使用dumpbin来查看程序的依赖
  • 原文地址:https://www.cnblogs.com/tommyli/p/1205078.html
Copyright © 2020-2023  润新知