• 18、(番外)匿名方法+lambda表达式


    概念了解

    1、什么是匿名委托(匿名方法的简单介绍、为什么要用匿名方法)

    2、匿名方法的【拉姆达表达式】方法定义

    3、匿名方法的调用(匿名方法的参数传递、使用过程中需要注意什么)

    什么是匿名方法?

    匿名方法是C#2.0引入的一个新特性,它允许开发者声明自己的函数代码而无须使用委托函数。

    C#为委托提供一种机制,可以为委托定义匿名方法,匿名方法没有名称,编译器会定指定一个名称,匿名方法中不能使用跳转语句跳转到该匿名方法的外部,也不能跳转到该方法的内部。也不能在匿名方法外部使用的ref和out参数。

    通过使用匿名方法,可以不必创建单独的方法,因此减少了实例化委托所需的编码系统开销。例如,如果创建方法所需的系统开销是不必要的,则指定代码块(而不是委托)可能非常有用。

    什么时候该使用匿名方法

    1.需要一个临时方法,该方法使用次数极少;
    2.这个方法的代码很短,甚至可能比方法声明都短的情况下使用。
     
    匿名方法的声明、调用
      

    C#3.0之后匿名方法可以使用λ表达式来进行定义

    无论是拉姆达(lambda)表达式(匿名函数、是匿名类) ,都能归属到一种叫闭包的东西上面。

    左边是参数,使用括号表达 (string param),可以是 (param)这样不定义类型,编译器会推断出来,只有一个参数的时候可以不使用括号。右边是实现代码,使用花括号,如果代码只有一行,则不使用花括号和return关键字也可以,编译器会为我们添加。

    应用:

    delegate string MyDelegate(string message);
     
    string str1 = "str1";
     
    string str2 = "str2";
     
    MyDelegate my1 = param => param + str1 + str2;
     
    MyDelegate my2 = (string param)=>{return param + str1 + str2;};
     
    Console.WreiteLine(my("param"))
     
    Console.WreiteLine(my2("param"));

    在我们程序中,经常有这样一些需求:

    1. 需要一个临时方法,这个方法只会使用一次,或者使用的很少。

    2. 这个方法的方法体很短,以至于比方法声明都短,写起来实在没劲(我将其称之为“一句话方法”)。

    没办法,这样的方法写起来真是吃力不讨好,比如一些按钮事件处理中,有些按钮点击就是弹出一个对话框,或者调用一下别的什么方法。比如下面的代码:

    this.btnRefresh.Click += new System.EventHandler(this.btnRefresh_Click);
    private void btnRefresh_Click(object sender, EventArgs e)
    {
        BindData();
    }

    这个”Refresh”按钮就是做一下调用一下BindData()数据绑定的方法,为此我们不得不写一个新方法。好了,C# 2.0为我们提供了匿名方法:

    this.btnRefresh.Click += delegate(object sender, EventArgs e) { BindData(); };

    没劲的代码没了。想知道这种写法的幕后黑手么?

    其实编译器还是在我们的后面干了一件龌龊的事情:它为我们产生了一个新的方法,它只是表面上为我们节省了代码。

    privatevoidb__0(object sender, EventArgs e)
    {
        this.BindData();
    }

    看看这个编译器产生的方法的名称:

    b_0,Test是这个匿名方法所放置的地方(因为这个按钮的时间我是放在一个Test方法里的) 还有一点需要注意的是,如果这个匿名方法是在实例方法里使用,那么编译器为我们生成的幕后方法也是实例方法,否则就是静态方法了。

    是不是觉得匿名方法这东西很不错,减少了很多代码阿,但是匿名方法的使用还并不人性化,什么是人性化呢?比如你可以用自然的语言将程序代码读出来,这样才算人性化了.在.net 2.0中System.Collections.Generic命名空间下List里有一些新增的方法。比如Find,如果使用匿名方法我们如何调用呢:

    books.Find(delegate(Book book){return book.Price < 50;});

    代码是很简单,但是却无法朗读出来,来看看Lambda表达式的写法:

    books.Find(book=>book.Price<50);这个Lambda表达式就可以这样阅读出来了:给你一本书,如果它的价格小于50则返回true。

    好了,那我们就走进Lambda表达式吧:

    将使用了Lambda表达式的程序集反编译后,我们发现,它实际上和匿名方法没有什么不同。Lambda的输入参数就对应着delegate括号里面的参数,由于Lambda表达式可以推断参数的类型,所以这里的参数无需声明。

    Lambda操作符读作”Goes to”,它后面紧跟着表达式或者是语句块(这点和匿名方法也不同,匿名方法只能使用语句块而不能使用表达式),下面我就用实例来说明一下有那些类型的Lambda表达式:

    //x的类型省略了,编译器可以根据上下文推断出来,后面跟着的是表达式

    //x的类型省略了,编译器可以根据上下文推断出来,后面跟着的是表达式
    x => x+1 
    deleage(int x){return x+1;}
    //后面跟着的是语句块
    x=>{return x+1;}
    delegate(int x){return x+1;}
    //输入参数也可以带类型,带类型后别忘记小括号哦
    (int x) => x+1
    delegate(int x){return x+1;}
    //也可以多个输入参数,逗号分隔,别忘记小括号
    (x,y) => x+y
    delegate(int x,int y){return x+y;}
    //无参的也行

    () => 1

    delegate(){return 1;}

    对于Lambda表达式来说她的用法就是如此,但是在Lambda背后却有很多的故事和玄机。用Lambda表达式可以构建表达式树,而表达式树对于Linq来说就像树根对于树一样重要。在这里就不讨论表达式树的问题了,这个东西也不是三言两语能够说清楚的,等待时机成熟的时候我们再来进一步讨论。

  • 相关阅读:
    【二分】Urozero Autumn Training Camp 2016 Day 5: NWERC-2016 Problem C. Careful Ascent
    【强连通分量缩点】【DFS】【动态规划】Urozero Autumn Training Camp 2016 Day 5: NWERC-2016 Problem B. British Menu
    【Splay】Codeforces Round #424 (Div. 1, rated, based on VK Cup Finals) B. Cards Sorting
    【Splay】bzoj3223 Tyvj 1729 文艺平衡树
    【Splay】bzoj3224 Tyvj 1728 普通平衡树
    【LIS】【递推】Gym
    【DFS】【枚举】Gym
    【递推】【DFS】【枚举】Gym
    【推导】Codeforces Round #424 (Div. 1, rated, based on VK Cup Finals) A. Office Keys
    【概率dp】【数学期望】Gym
  • 原文地址:https://www.cnblogs.com/wleaves/p/5261203.html
Copyright © 2020-2023  润新知