用面向对象思维理解.NET委托:函数是对象、委托是函数对象的类型
2008-09-03 14:24
357 查看
(原创文章,转载请注明来源:http://blog.csdn.net/hulihui)
不谈C/C++,就从对象和类型考虑.NET委托(delegate),笔者的看法是:函数是对象、委托是函数对象的类型。
教科书上讲:具有同属性的实体抽象成类型。那么,程序代码中函数的属性是啥?就是函数签名,即函数返回类型、函数参数类型。进一步,具有相同属性的函数抽象成啥呢?就是.NET中所谓的委托(delegate),它就是函数的类型。与普通对象与类型不同,委托的对象(函数)可以不要new操作符号创建,因为函数已经存在于代码中。委托自己没有成员,它只是一类函数的代表而已,所以叫做delegate,也可翻译成代表。
举一个把函数作为参数调用的例子。
int fun2(int n)
[align=left] {
[/align] 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++、函数指针、函数回调、调用安全、事件机制等方面讲委托。事实上,从面向对象的观点看,函数是实体,那么就可认为是对象,再抽象就成函数类型——委托了。这样想,许多原先模糊的观念就清晰多了。
不谈C/C++,就从对象和类型考虑.NET委托(delegate),笔者的看法是:函数是对象、委托是函数对象的类型。
教科书上讲:具有同属性的实体抽象成类型。那么,程序代码中函数的属性是啥?就是函数签名,即函数返回类型、函数参数类型。进一步,具有相同属性的函数抽象成啥呢?就是.NET中所谓的委托(delegate),它就是函数的类型。与普通对象与类型不同,委托的对象(函数)可以不要new操作符号创建,因为函数已经存在于代码中。委托自己没有成员,它只是一类函数的代表而已,所以叫做delegate,也可翻译成代表。
举一个把函数作为参数调用的例子。
int fun2(int n)
[align=left] {
[/align] 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++、函数指针、函数回调、调用安全、事件机制等方面讲委托。事实上,从面向对象的观点看,函数是实体,那么就可认为是对象,再抽象就成函数类型——委托了。这样想,许多原先模糊的观念就清晰多了。
相关文章推荐
- 用面向对象思维理解.NET委托:函数是对象、委托是函数对象的类型
- 用面向对象思维理解.NET委托:函数是对象、委托是函数对象的类型
- 简单理解面向对象思维
- 真的理解面向对象思维
- 类型兼容性原则遇上还是函数重写_面向对象新需求
- 对c++中模板函数函数类型或者函数对象了类型显式指定的理解
- java面向对象思维的理解
- PHP面向对象深入理解之五(内省函数与反射类)
- C++ Pirmer : 第十五章 : 面向对象程序设计之基类和派生的定义、类型转换与继承与虚函数
- C++11 理解 (二十八) 之 用于计算函数对象返回类型的统一方法
- 【面向对象程序设计常见面试题】数组在做函数实参的时候会转变为什么类型?
- 面向对象程序设计上机练习二(函数模板)
- JavaScript总体的介绍【JavaScript介绍、定义函数方式、对象类型、变量类型】
- 【C++】面向对象类成员函数的重载、覆盖与隐藏
- 面向对象_多态的成员访问特点级转型的理解
- 面向对象和面向过程的理解
- 引用数据类型 类 封装 类与对象 面向对象 私有private this关键字
- 【对象数组+静态数据成员+静态成员函数+...】面向对象程序设计(B)——第二次作业
- 面向对象思维举例
- 面向对象的程序设计1 理解对象—— JS学习笔记2015-7-4(第75天)