澳门至尊网站-首页

您的位置:澳门至尊网站 > 技术教程 > 还大概有Lambda的涉及和区分

还大概有Lambda的涉及和区分

2019-11-08 06:30

以前一直迷迷糊糊的,现在总算搞明白。

一、简介

  委托是一种类型,由关键字delegate声明。确切的说,委托是一种可用于封装命名或者匿名方法的引用类型。  它类似于 C++ 中的函数指针,而且是类型安全和可靠的。

  委托类型的声明与方法签名相似,有一个返回值和任意数目任意类型的参数。必须使用具有兼容返回类型和输入参数的方法或 lambda 表达式实例化委托。

  委托允许将方法作为参数进行传递。
  委托可用于定义回调方法。
  委托可以链接在一起;例如,可以对一个事件调用多个方法。
  方法不必与委托签名完全匹配。

Lambda表达式

二、Delegate

   Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型。

     public delegate int MethodDelegate(int x, int y);
        private static MethodDelegate method;
        static void Main(string[] args)
        {
            method = new MethodDelegate(Add);
            Console.WriteLine(method(10,20));
            Console.ReadKey();
        }

        private static int Add(int x, int y)
        {
            return x + y;
        }

 

Lamda表达式基本写法是()=>{ };
Lambda和方法一样都可以传入参数和拥有返回值。(int x)=>{return x;};
Lambda表达式并不是C#独有的,其它语言也拥有Lambda表达式。

三、Func<T>

 在使用 Func<T, TResult> 委托时,不必显式定义一个封装只有一个参数的方法的委托。

 以下示例简化了此代码,它所用的方法是实例化 Func<T, TResult> 委托,而不是显式定义一个新委托并将命名方法分配给该委托。

public class GenericFunc
{
   public static void Main()
   {
      // 依旧是用命名方法实例化委托类型
      Func<string, string> convertMethod = UppercaseString;
      string name = "Dakota";
      // 依旧是通过委托实例调用该方法
      Console.WriteLine(convertMethod(name));
   }

   private static string UppercaseString(string inputString)
   {
      return inputString.ToUpper();
   }
}

下面的示例演示如何声明和使用 Func<T, TResult> 委托。  

此示例声明一个 Func<T, TResult> 变量,并为其分配了一个将字符串中的字符转换为大写的 lambda 表达式。  

随后将封装此方法的委托传递给Enumerable.Select 方法,以将字符串数组中的字符串更改为大写。

static class Func
{
   static void Main(string[] args)
   {
      // 声明了一个Func委托类型的变量selector并用Lambda表达式进行实例化 
      // 这个Lambda表达式将用来获取一个字符串并将这个字符串转化为大写并返回
      Func<string, string> selector = str => str.ToUpper();

      // 创建一个字符串数组
      string[] words = { "orange", "apple", "Article", "elephant" };
      // 依次遍历这个字符串数组并调用委托实例selector进行处理
      IEnumerable<String> aWords = words.Select(selector);

      // 输出结果到控制台
      foreach (String word in aWords)
         Console.WriteLine(word);
   }
}      
/*
This code example produces the following output:

   ORANGE
   APPLE
   ARTICLE
   ELEPHANT
*/

 

一些文档上写Lambda是匿名函数,我认为是不对的,Lambda可以表示一个匿名函数,但是还可以来表示Linq表达式啊。

四、Action<T>

Action 委托:没有传入参数,也没有返回类型,即Void。如:

void Main(string[] args)
        {
            Action say = SayHello;
            say();
        }
        public static void SayHello( )
        {
           Console.WriteLine("Say Hello");
        }

Action<T> 委托:传入参数为T,没有返回类型。如:

void Main(string[] args)
        {
            Action<string> say = SayHello;
            say("Hello");
        }
        public static void SayHello(string word )
        {
            Console.WriteLine(word);
        }

其实ActionFunc的用法差不多,差别只是一个有返回类型,一个没有返回类型,当然Action也可以接匿名方法和Lambda表达式。

匿名方法:

void Main(string[] args)
        {
            Action<string> say = delegate(string word)
            {
                Console.WriteLine(word);
            };
            say("Hello Word");
        }

Lambda表达式:

 static void Main(string[] args)
        {
            Action<string> say = s => Console.WriteLine(s);
            say("Hello Word");
        }

 

delegate委托

五、Predicate<T>

 泛型委托:表示定义一组条件并确定指定对象是否符合这些条件的方法。此委托由 Array 和 List 类的几种方法使用,用于在集合中搜索元素。

void Main(string[] args)
        {
            Point[] points = { new Point(100, 200), 
            new Point(150, 250), new Point(250, 375), 
            new Point(275, 395), new Point(295, 450) };
            Point first = Array.Find(points, ProductGT10);
            Console.WriteLine("Found: X = {0}, Y = {1}", first.X, first.Y);
            Console.ReadKey();
        }
        private static bool ProductGT10(Point p)
        {
            if (p.X * p.Y > 100000)
            {
                return true;
            }
            else
            {
                return false;
            }
        }

使用带有 Array.Find 方法的 Predicate 委托搜索 Point 结构的数组。

如果 X 和 Y 字段的乘积大于 100,000,此委托表示的方法 ProductGT10 将返回 true。

Find 方法为数组的每个元素调用此委托,在符合测试条件的第一个点处停止。

委托基本的使用包括声明一个委托、实例化一个委托、使用一个委托

六、总结

    Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型

  Func可以接受0个至16个传入参数,必须具有返回值

  Action可以接受0个至16个传入参数,无返回值

  Predicate只能接受一个传入参数,返回值为bool类型

 

参考文章:

// 声明一个委托,类型是myDelegate,
public delegate void myDelegate(string str); 
public static void HellowChinese(string strChinese) 
{ 
    Console.WriteLine("Good morning," + strChinese); 
    Console.ReadLine(); 
} 
// 实例化   
myDelegate d = new myDelegate(HellowChinese); 
// 使用
d("Mr wang");

委托很好用。当编程到一定阶段,发现只传递一个或几个int、strig类型参数是不够的,希望可以把一段代码进行传递来执行某个操作,委托提供了这样的途径,委托提供了一种可以直接访问类中方法的途径,可以将方法当作一个参数传递从而使用

匿名函数

上面已经讲到匿名函数了。函数和方法等价,用来封装一段代码片以便随时使用,有的有返回值有的无返回值。写在类里面。

但是如果我只想把这段代码片使用一次呢?再声明一个类、一个方法实在太麻烦,这时候匿名函数就拥有了价值。用来封装一段代码、直接执行或者传递。

匿名函数总是和委托一齐使用,因为匿名函数方便了委托的使用(不用声明一个类了)

匿名方法两种实现方式:

使用Lambda表达式:

        public delegate void Seep();
        static void Main(string[] args)
        {

            Seep s = () => { Console.WriteLine(2); Console.WriteLine(1); };
        }

使用委托

 

        public delegate void Seep();
        static void Main(string[] args)
        {
            Seep see = delegate () { Console.WriteLine(1); };
        }

 

Action、Func、Predicate

委托需要先声明一个类型才能使用,这很麻烦,比如我在某一个命名空间声明了一个没有返回值的委托myDelegate,然后想要这个委托的时候,每个命名空间都要引用这个命名空间,这太不友好了。

微软直接就在框架中定义了三种委托:Action、Func、Predicate。这三种委托是C#的System命名空间中已经定义好的,如Action

图片 1

上面已经通过匿名函数实现了不用声明类,现在通过C#预定义的委托类型实现了不用声明委托类型,那么现在就可以直接写代码片来执行了

这三种委托的相应泛型形式,可以写出如下代码:

            #region 无参数无返回值
            // lambda实现委托实例
            Action a1 = () => Console.WriteLine(1);
            a1();
            // delegate实现委托实例,下面不再演示delegate。
            Action a2 = delegate { Console.WriteLine(1); };
            a2();
            #endregion

            #region 有参数无返回值
            Action<int> a3 = (x) => Console.WriteLine(x);
            a3(1);
            #endregion

            #region 无参数有返回值的情况
            Func<int> e1= ()=> 1;
            var value1= e1();
            #endregion

            #region 有参数有返回值的情况
            // 最后一个泛型是返回值的类型
            Func<int,int> e2 = (x) => x+1;
            int value2= e2(1);
            // 多参数情况
            Func<int, int,int> e3 = (x,x2) => x+x2 + 1;
            int value3 = e3(1,2);
            #endregion

            #region 返回值是布尔类型
            // 其实Predicate可以用Func返回值是bool来代替的。
            Predicate<int> p = (x) => x > 0;
            bool value4 = p(1);
            #endregion

委托还有一些特性(如可加性),感兴趣的可以查阅相关资料。

Expression

Expression指的是System.Linq.Expressions,是Linq表达式,表达式树,不是Lambda表达式!!

Linq表达式用于离线集合的查询。什么是离线集合?最常用的是EF里面的DBContext,我也只用过这个,所以我不想那么多,就把Expression当成EF里面使用的就行了。

1、只能接受Lambda表达式表示的委托

2、Lambda表达式只能有一条语句。

            Expression<Func<int,int>> eeee = (x)=> x+ 1 ;

总结

1、都是希望可以单独引用一段代码片引起的。
2、然后引用了委托delegate,delegate的使用方式为声明类型-实例化类型(传入想要使用的类的方法)-使用委托(使用方法)
3、简化操作,使用(通过Lambda或者delegate)匿名方法来直接声明一段代码片,避免声明所想使用的类的方法
4、简化操作,直接使用C#预定义的三种委托Action、Func、Predicate,连声明委托都省了。
5、本文代码:

注意

1、匿名函数总是和委托一起使用(三种预定义委托)
2、使用三种预定义委托就可以满足大多数需求。
3、Expression是Linq表达式树,只能使用接受一句lambda表达式。和匿名函数无关。

本文由澳门至尊网站发布于技术教程,转载请注明出处:还大概有Lambda的涉及和区分

关键词:

  • 上一篇:没有了
  • 下一篇:没有了