技术开发 频道

一次性搞定C#委托、匿名和Lambda表达式

    【IT168 评论】在.NET中,委托,匿名方法和Lambda表达式很容易发生混淆。我想下面的代码能证实这点。下面哪一个First会被编译?哪一个会返回我们需要的结果?即Customer.ID=5.答案是6个First不仅被编译,并都获得正确答案,且他们的结果一样。如果你对此感到困惑,那么请继续看这篇文章。

class Customer
{
    
public int ID { get; set; }
    
public static bool Test(Customer x)
    {
        return x.ID
== 5;
    }
}
...
List
<Customer> custs = new List<Customer>();
custs.Add(
new Customer() { ID = 1 });
custs.Add(
new Customer() { ID = 5 });

custs.First(
new Func<Customer, bool>(delegate(Customer x) { return x.ID == 5; }));
custs.First(
new Func<Customer, bool>((Customer x) => x.ID == 5));
custs.First(delegate(Customer x) { return x.ID
== 5; });
custs.First((Customer x)
=> x.ID == 5);
custs.First(x
=> x.ID == 5);
custs.First(Customer.Test);

      什么是委托?

  现在你定义一个处理用户订单的购物车ShoppingCart类。管理层决定根据数量,价格等给客人折扣。做为其中的一部分,他们已经实现了处理订单时你要考虑一方面。不用考虑过多,你简单声明一个变量来保存有“吸引力的折扣”(magicDisCount),然后实现逻辑。

class Program {
    static void Main(
string[] args)  {
        
new ShoppingCart().Process();
    }
}  
class ShoppingCart {
    
public void Process() {
        
int magicDiscount = 5;
        
// ...
    }
}

 

  第二天,异想天开的管理层决定根据购买时间调整折扣。这个很简单,但需要你改动一点代码。

class ShoppingCart {
    
public void Process() {
        
int magicDiscount = 5;
        
if (DateTime.Now.Hour < 12) {
            magicDiscount
= 10;
        }
    }
}

 

  接下来一段时间里,管理层又反复添加更多的折扣逻辑。这时你就会在心理抱怨“受够了”。那么我该怎么做才能把这些无聊的逻辑从我的代码中剥离出去,让该处理的人去处理呢?这时你要做的是移交或者委派给相应职能的别人。幸运的是,.NET为此提供了一种叫做“委托”的机制。

  委托

  如果你有C/C++编程背景,描述委托最好的方法是“函数指针”。对所有人来说,可以认为把委托传递给方法与把值或对象传递给方法一样。比如下面三行代码就表现出一样的基本原则:你在传递数据给Process处理而不是你自己使用。

// 给方法Process传递一个整形值
Process(
5 );
// 给方法Process传递一个ArrayList的引用
Process(
new ArrayList() );
// 给方法Process传递一个方法的引用
Process( discountDelegate );

 

  DiscountDelegate是什么?我如何创建?Process方法如何使用?首先如同声明一个类一样,声明一个委托类型。

delegate int DiscountDelegate();

 

  这句话的意思是我们有一个叫DiscountDelegate的委托类型,我们可以像使用类,结构体等一样使用它。它不需要数据参数,但返回一个整数值。像类一样,我们必须创建一个它的实例它才有意义。记住,创建一个委托实例实质上是创建一个方法的引用。创建实例时关键是要明白DiscountDelegate没有任何构造器,它有一个隐式的构造函数来构造一个与它相同签名的方法(没有传入参数,返回一个整数)。那你怎么给这个构造函数一个方法呢?.NET向你提供了一个向它名字一样简单的方法,你所做的只是忽略圆括号。

DiscountDelegate discount = new DiscountDelegate(class.method);

 

  在深入之前,先回到开始的例子,整理一个代码。我们会添加一个Calculator类来帮助我们处理折扣逻辑,并给我们的委托提供一些方法。

delegate int DiscountDelegate();

class Program {
    static void Main(
string[] args) {
        Calculator calc
= new Calculator();
        DiscountDelegate discount
= null;
        
if (DateTime.Now.Hour < 12) {
            discount
= new DiscountDelegate(calc.Morning);
        }
        
else if (DateTime.Now.Hour < 20) {
            discount
= new DiscountDelegate(calc.Afternoon);
        }
        
else {
            discount
= new DiscountDelegate(calc.Night);
        }
        
new ShoppingCart().Process(discount);
    }
}  
class Calculator {
    
public int Morning() {
        return
5;
    }
    
public int Afternoon() {
        return
10;
    }
    
public int Night() {
        return
15;
    }
}  
class ShoppingCart {
    
public void Process(DiscountDelegate discount) {
        
int magicDiscount = discount();
        
// ...
    }
}

 

0
相关文章