用面向对象思维理解.NET委托:函数是对象、委托是函数对象的类型
2015-09-25 11:42
357 查看
不谈C/C++,就从对象和类型考虑.NET委托(delegate),笔者的看法是:函数是对象、委托是函数对象的类型。
教科书上讲:具有同属性的实体抽象成类型。那么,程序代码中函数的属性是啥?就是函数签名,即函数返回类型、函数参数类型。进一步,具有相同属性的函数抽象成啥呢?就是.NET中所谓的委托(delegate),它就是函数的类型。与普通对象与类型不同,委托的对象(函数)可以不要new操作符号创建,因为函数已经存在于代码中。委托自己没有成员,它只是一类函数的代表而已,所以叫做delegate,也可翻译成代表。
举一个把函数作为参数调用的例子。
int fun2(int n)
{
int m = 0;
for(int k = 1; k <= n; k++)
{
m += k * k;
}
return m
}
int fun3(int n)
{
int m = 0;
for(int k = 1; k <= n; k++)
{
m += k * k * k;
}
return m
}
函数fun2计算1到n的平方和,函数fun3计算1到n的立方和。它们的属性相同:返回int、参数int。于是定义这两个函数的类型——一个委托TSum如下。注意,TSum是类型(Type),其返回类型、参数类型与fun2、fun3相同。
public delegate int TSum(int n);
这样,就可以定义调用函数的函数output(),如:
void output(TSum sum)
{
MessageBox.Show(sum(10).ToString());
}
最后,用 output(fun2)、output(fun3),将显示10以内平方数和、立方数和。当然,如果完全按照面向对象方式,可以先创建两个TSum实例,然后调用,如:
TSum f2 = new TSum(fun2);
TSum f3 = new TSum(fun3);
output(f2);
output(f3);
后一种方式产生的f2、f3在内部机制上与output(fun2)、output(fun3)有较大的区别,详细阐述见Jeffrey Richter的名著《框架设计(第2版:CLR Via C#》。
.NET事件机制也可以理解:事件是函数、委托规定了函数的属性。在激发事件时,必须判断事件是否注册,即实现如下类似代码:
if(OnClickEvent != null) // 相当于判断 OnClickEvent 对象是否存在
{
OnClickEvent(this, new ClicktEventArgs(...));
}
刚接触.NET时,总觉得委托delegate难以理解,笔者认为关键原因是:没有从面向对象的观点上考虑函数与委托的关系,而是从传统的C/C++、函数指针、函数回调、调用安全、事件机制等方面讲委托。事实上,从面向对象的观点看,函数是实体,那么就可认为是对象,再抽象就成函数类型——委托了。这样想,许多原先模糊的观念就清晰多了。
http://blog.csdn.net/hulihui/article/details/2873519
教科书上讲:具有同属性的实体抽象成类型。那么,程序代码中函数的属性是啥?就是函数签名,即函数返回类型、函数参数类型。进一步,具有相同属性的函数抽象成啥呢?就是.NET中所谓的委托(delegate),它就是函数的类型。与普通对象与类型不同,委托的对象(函数)可以不要new操作符号创建,因为函数已经存在于代码中。委托自己没有成员,它只是一类函数的代表而已,所以叫做delegate,也可翻译成代表。
举一个把函数作为参数调用的例子。
int fun2(int n)
{
int m = 0;
for(int k = 1; k <= n; k++)
{
m += k * k;
}
return m
}
int fun3(int n)
{
int m = 0;
for(int k = 1; k <= n; k++)
{
m += k * k * k;
}
return m
}
函数fun2计算1到n的平方和,函数fun3计算1到n的立方和。它们的属性相同:返回int、参数int。于是定义这两个函数的类型——一个委托TSum如下。注意,TSum是类型(Type),其返回类型、参数类型与fun2、fun3相同。
public delegate int TSum(int n);
这样,就可以定义调用函数的函数output(),如:
void output(TSum sum)
{
MessageBox.Show(sum(10).ToString());
}
最后,用 output(fun2)、output(fun3),将显示10以内平方数和、立方数和。当然,如果完全按照面向对象方式,可以先创建两个TSum实例,然后调用,如:
TSum f2 = new TSum(fun2);
TSum f3 = new TSum(fun3);
output(f2);
output(f3);
后一种方式产生的f2、f3在内部机制上与output(fun2)、output(fun3)有较大的区别,详细阐述见Jeffrey Richter的名著《框架设计(第2版:CLR Via C#》。
.NET事件机制也可以理解:事件是函数、委托规定了函数的属性。在激发事件时,必须判断事件是否注册,即实现如下类似代码:
if(OnClickEvent != null) // 相当于判断 OnClickEvent 对象是否存在
{
OnClickEvent(this, new ClicktEventArgs(...));
}
刚接触.NET时,总觉得委托delegate难以理解,笔者认为关键原因是:没有从面向对象的观点上考虑函数与委托的关系,而是从传统的C/C++、函数指针、函数回调、调用安全、事件机制等方面讲委托。事实上,从面向对象的观点看,函数是实体,那么就可认为是对象,再抽象就成函数类型——委托了。这样想,许多原先模糊的观念就清晰多了。
http://blog.csdn.net/hulihui/article/details/2873519
相关文章推荐
- 用户登录中Cookie的简单应用
- 二维码
- chrom浏览器模拟手机访问
- 反射-框架、工具类
- hive导出查询结果到本地文件
- Java中的Callback 回调函数
- mysql 事物小测试
- 黑马程序员---Java基础加强---JDK1.5新特性
- 处理千万级以上的数据提高查询速度的方法 ...
- PRML第一章_易混淆概念-先验后验&生成判别&分类回归
- Ionic开发环境搭建
- [经典算法] 排列组合-N元素集合的M元素子集
- 使用flume问题总结1——搭建flume+测试Syslog source
- codeforces 580D Kefa and Dishes
- classLoader和javassist
- 算法之每日一题:哥德巴赫猜想
- OWC 绘制3D柱状图
- git之 git checkout
- 网站渗透测试
- Android中Application、静态变量和Sharedpreferences的使用与区别