本文为原创,如需转载,请注明作者和出处,谢谢!
在C#2.0及C#1.x中,需要使用delegate来定义方法指针。如下面的代码如示:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
public
delegate
bool
Filter(
int
num);
//
delegate类型
public
int
[]searchArray(
int
[]values,Filterfilter)
{
List
<
int
>
result
=
new
List
<
int
>
();
foreach
(
int
i
in
values)
{
if
(filter(i))
result.Add(i);
}
return
result.ToArray();
}
从上面的代码可以看出,定义了一个叫Filter的delegate类型。然后searchArray方法的第二个参数通过Filter类型将方法指针传到该方法中。 实际上,从IL可以看出,Filter被编译成了一个类,而且这个类是System.MulticastDelegate的子类。但和普通类不同的是,System.MulticastDelegate及其子类只能由编译器使用,不能直接写在源程序中。也就是说,只能在IL中使用。
如果要调用searchArray方法,需要先定义一个和Filter拥有同样参数和返回值的方法,代码如下:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
public
bool
MyFilter(
int
num)
{
return
((num
&
1
)
==
0
);
}
下面的代码调用了searchArray方法:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
int
[]intArray
=
new
int
[]{
1
,
2
,
3
,
4
,
5
,
6
,
7
,
8
};
int
[]evenArray
=
searchArray(intArray,MyFilter);
foreach
(
int
i
in
evenArray)
textBox1.AppendText(i.ToString());
从C#2.0开始支持匿名方法,开发人员可以通过匿名方法用内联代码形式取代delegate,如上面的调用代码可以修改为下面的形式:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
int
[]intArray
=
new
int
[]{
1
,
2
,
3
,
4
,
5
,
6
,
7
,
8
};
int
[]evenArray
=
searchArray(intArray,
delegate
(
int
i){
return
((i
&
1
)
==
0
);});
foreach
(
int
i
in
evenArray)
textBox1.AppendText(i.ToString());
上面的代码将匿名方法转换成了delegate。
然而,从C#3.0(.net framework3.5)开始,支持了Lambda表达式。所谓Lambda表达式就是delegate和匿名方法的简写形式,Lambda表达式的语法如下:
(param1, param2 ...,paramN) =>
{
表达式1;
表达式2;
return 返回值;
}
上面语法中的param1...paramN就表示方法的参数(不用确定类型,C#编译器会为我们做这个工作),而{...}里面的内容就和方法体中的内容完全一样。
如果delegate没有参数,可以只写(),如下面的方法所示:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
public
delegate
void
Method1();
public
void
test()
{
Method1method1
=
()
=>
{
int
i
=
4
;i
+=
6
;};
}
如果delegate只有一个参数,参数两边的括号可以不写,代码如下:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
public
delegate
void
Method2(
int
i);
public
void
test()
{
Method2method2
=
i
=>
{i
++
;i
+=
6
;};
}
如果delegate有返回值,{...}中的最后一条语句需要使用return来返回相应的值,代码如下:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
public
delegate
int
Method3(
int
x,
int
y);
public
void
test()
{
Method3method3
=
(x,y)
=>
{x
++
;y
++
;
return
x
+
y;};
}
如果使用Lambda表达式改写本文开头的例子,就会变得非常简单,代码如下:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
int
[]intArray
=
new
int
[]{
1
,
2
,
3
,
4
,
5
,
6
,
7
,
8
};
int
[]evenArray
=
searchArray(intArray,i
=>
{
return
(i
&
1
)
==
0
;});
foreach
(
int
i
in
evenArray)
textBox1.AppendText(i.ToString());
当然,我们也可以将Lamdba表达式赋给一个delegate,再进行传值,代码如下;
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
Filterfilter
=
i
=>
{
return
(i
&
1
)
==
0
;};
int
[]intArray
=
new
int
[]{
1
,
2
,
3
,
4
,
5
,
6
,
7
,
8
};
int
[]evenArray
=
searchArray(intArray,filter);
foreach
(
int
i
in
evenArray)
textBox1.AppendText(i.ToString());
从本质上讲,Lamdba表达式经过C#编译器编译后,仍然会变成delegate的形式,也就是说Lamdba表达式只是在语法层次上的改进,并不是IL提供的新的指令。如下面的两行代码是等价的:
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
Filterfilter
=
i
=>
{
return
(i
&
1
)
==
0
;};
Filterfilter
=
delegate
(
int
i){
return
((i
&
1
)
==
0
);};