JavaScript中的函数和C#中的匿名函数,委托、lambda表达式

在js中function是一个一个引用类型,所以可以出现这样的代码:

'use strict';
var compare=function(value1, value2) {
    if (value1<value2) {
        return -1;
    }
    else if (value1>value2) {
        return 1;
    } else {
        return 0;
    }
}
var arr = [1, 2, 4, 6, 3, 4];
arr.sort(compare);//可以直接传递一个函数名给这个方法作为参数。
console.log(arr); 

这种在js里面很平常的用法,在c#里面是不被允许的,在c#中方法是一个类中存放数据处理逻辑的地方,谈不上是一个引用类型,在c#中相似的用法是委托。看下面的例子:

 public delegate void DoSomeThingElse(int number);
        static void Main(string[] args)
        {
            DoSomeThingElse delegateDo = null;
            FuncDoSomeThingElse(delegateDo);//主要是这里,实现了和js一样的功能,可以直接传入一个方法体
            Console.ReadKey();
        }

        public static void FuncDoSomeThingElse(DoSomeThingElse toDo)
        {
            var number = 1;
            toDo += Person.DoSomeThing;
            toDo(number);
        }

        
    }

    public class Person
    {
        public static void DoSomeThing(int number)
        {
            Console.WriteLine(number);
        }
    }

可以看出如果要在C#中使用与js中一样的功能是非常困难的,首先要定义一个方法,在这个例子中是在person这个类中定义了一个DoSomeThing的方法,接受一个number参数。第二步是定义一个方法去接受一个委托的方法(

FuncDoSomeThingElse),这方法中对委托进行操作,这个例子中是利用toDo += Person.DoSomeThing;这条语句在内部调用delegate的combine方法将person的DoSomeThing方法加入到了委托链中。接下来在main方法中对委托进行实例化.....总之心很累。
C#3中加入了匿名委托:
class Program
    {
        public delegate void DoSomeThingElse(int number);
        static void Main(string[] args)
        {
            DoSomeThingElse dooo = delegate (int shit)
              {
                  Console.WriteLine(shit);
              };
            dooo(5);
            
            Console.ReadKey();
        }

       


    }

看一下已经变得清爽了许多,不需要在因为一个方法而去创建一个类来包容了,直接一个delegate(){}的表达式来搞定。但是这样的写法还是阅读起来不是很顺畅,虽然已经清爽了很多,于是在C#3.5\C#4中添加的lambda表达式使得这种行为更加的简单了:

 class Program
    {
       
        static void Main(string[] args)
        {
            Action<int> dooo =a=> Console.WriteLine(a);
            dooo(5);
            
            Console.ReadKey();
        }    
    }

是不是爽爆了?这里出现的新东西是Action和=>,Action表示一个内建的委托,还有一个是Func,还有一个是Predicate,表达的意思不一样,但是这三种内建的委托基本能够代表所有的委托类型了。所以,连delegate这个东西都可以丢掉了。=>表达的意思是这是一个lambda表达式,在这个符号的左边是传入的参数,如果有多个参数的话可以用圆括号括起来。右边是方法体,如果有多行语句的话还可以在这个符号的右边加上一个花括号。

lambda表达式这个东西是编译器帮助我们做了很多的事情,在后台,编译器为方法体建立一个包容的类来放置这个方法体,闭包也和这个东西有关系,闭包直观的说就是因为匿名函数引用了局部变量而引起的一种行为。这个是后话。