1.为什么要使用委托?
生活中的委托就是委托他人帮我们去办一件事情,程序中的委托类似。看下面的例子
class Class1
{
static void Main(String[] args)
{
List<int> list = new List<int>();
list.Add();
list.Add();
list.Add();
list.Add();
list.Add(); //过滤偶数
Even(list); ObjectDumper.Write(list);
} //过滤集合中的偶数
public static void Even(List<int> list)
{
for (int i = ; i < list.Count; i++)
{
if (list[i] % 2 != )
{
list.RemoveAt(i);
i--;
}
}
} //过滤集合中的奇数
public static void Odd(List<int> list)
{
for (int i = ; i < list.Count; i++)
{
if (list[i] % 2 == )
{
list.RemoveAt(i);
i--;
}
}
}
}
通过上面的例子我们发现 Even()方法 和 Odd()方法只有一行代码不同,使用委托可以将方法当成参数传递,这样做的好处是使程序之间的耦合降低,同时节省代码。
改造上面的代码如下:
class Class2
{
//声明委托
public delegate bool Cal(int num);
static void Main(String[] args)
{
List<int> list = new List<int>();
list.Add();
list.Add();
list.Add();
list.Add();
list.Add(); //过滤偶数
Cal cal = new Cal(Even);
MyCal(list, cal); ObjectDumper.Write(list);
} public static bool Even(int num)
{
if (num % == )
{
return true;
}
return false;
}
public static bool Odd(int num)
{
if (num % != )
{
return true;
}
return false;
} public static void MyCal(List<int> list,Cal cal)
{
for (int i = ; i < list.Count; i++)
{
if (cal(list[i]))
{
list.RemoveAt(i);
i--;
}
}
} }
委托描述的是方法的签名(包括参数类型、个数、返回值),上面声明一个委托 Cal ,和 我们要调用的方法一样,接收一个 int类型的参数,返回bool类型。
public delegate bool Cal(int num);
使用New关键字创建委托:
Cal cal = new Cal(Even);
最后调用cal,得到所有偶数项的集合
MyCal(list, cal);
我们在 MyCal(List<int> list,Cal cal) 方法中,传递了一个委托 作为参数,Cal描述的是 Even 或者Odd方法的签名,这样可以在调用的时候决定 是过滤奇数 还是 过滤偶数。
这和我们直接调用Even或Odd方法有什么不同,不是更麻烦了吗,接着往下看:
我们可以将过滤的方法改成如下代码:
MyCal(list, delegate(int i)
{
if (i % == )
{
return true;
}
return false;
});
正因为MyCal方法接收的是一个委托,所以我们可以在不定义Even或Odd的时候,使用匿名委托。
这样省去了不必要的代码,使结构看起来更简单,也不需要预先定义好方法,在调用的时候我们临时决定。
.NET 还给我们提供了更进一步的简化,使用Lambda表达式:
//代码演进
//lamada
MyCal(list, (i) => { return i % == ? true : false; });
i表示输入参数 (i) => { return i % 2 == 0 ? true : false; } 是一条lambda语句,标识过滤偶数。
完整的代码如下:
//声明委托
public delegate bool Cal(int num);
static void Main(String[] args)
{
List<int> list = new List<int>();
list.Add();
list.Add();
list.Add();
list.Add();
list.Add(); MyCal(list, (i) => { return i % 2 == 0 ? true : false; }); ObjectDumper.Write(list);
} public static void MyCal(List<int> list, Cal cal)
{
for (int i = ; i < list.Count; i++)
{
if (cal(list[i]))
{
list.RemoveAt(i);
i--;
}
}
}
这样做是不是比开始的时候简单多了,无论后面再怎么添加新的过滤方式,我们只需要修改我们的lambda语句,其他的都不需要变动,这就是委托的魅力。