C 函数指针与C
创始人
2024-06-22 08:30:36
0

 C#中有一种应用方法叫做委托。其应用方式及实现功能和我们以前为大家介绍的C++函数指针有不少相同之处。在这篇文章中,我们将会针对这两种方法做一个简单的介绍,方便大家区分理解。

委托和函数指针都描述了方法/函数的签名,并通过统一的接口调用不同的实现。但二者又有明显的区别,简单说来,委托对象是真正的对象,而函数指针变量只是函数的入口地址。对于高层应用来讲,委托的灵活性和适用范围更胜C++函数指针;但对于底层应用,函数指针则是不可替代的。下面分别是委托类型和函数指针类型定义的例子:

  1. delegate int Fn(int a, int b) //C#委托  
  2. typedef int (*Fn)(int a, int b) //C++函数指针 

从形式上看,二者的参数列表和返回值都是相同的,只是一个采用关键字delegate,一个采用指针符号*。似乎“相似”的说法更有把握了,但如果马上给二者划等号就操之过急了。我们先实际验证一下,看看到底有什么不同:

  1. //C#  
  2. delegate int Fn(int a, int b) ;  
  3. class Adder{  
  4. private int c = 0;  
  5. public int Add(int a, int b){  
  6. return a + b + c;  
  7. }  
  8. public Adder(int c){ this.c = c; }  
  9. }  
  10. class Multiplier{  
  11. private int c = 0;  
  12. public int Multiple(int a, int b){  
  13. return a * b * c;  
  14. }  
  15. public Multiplier(int c){ this.c = c; }  
  16. }  
  17. Adder adder = new Adder(1);  
  18. Multiplier multiplier = new Multiplier(2);  
  19. Fn fn = adder.Add;  
  20. fn(1, 2); //结果为4  
  21. fn = multiplier.Multiple;  
  22. fn(2, 3); //结果为12 

从上面的代码说明了两个问题:

1.委托对象可以指向不同类的方法,只要符合委托签名;

2.委托对象是有状态的(保存在指向的对象中),委托的行为不仅受到输入参数的影响,还受到目标对象状态的影响。

  1. //C++  
  2. typedef int(*Fn)(int a, int b);   
  3. int Add(int a, int b) {   
  4. return a + b;   
  5. };   
  6. int Multiple(int a, int b) {   
  7. return a * b;   
  8. };   
  9. class Adder {   
  10. public:   
  11. Adder(int c) {   
  12. this->cc = c;   
  13. }   
  14. int Add(int a, int b) {   
  15. return a + b + c;   
  16. }   
  17. private:   
  18. int c;   
  19. };   
  20. typedef int(Adder::* Fm)(int a, int b);   
  21. int _tmain(int argc, _TCHAR* argv[])   
  22. {   
  23. Fn fn = Add;   
  24. std::cout << fn(1, 2) << std::endl;   
  25. fn = Multiple;   
  26. std::cout << fn(1, 2) << std::endl;   
  27. Adder adder(1);   
  28. Fm f = &Adder::Add;   
  29. std::cout << (adder.*f)(1, 2) << std::endl;   
  30. return 0;   

C#中的委托是一种支持()操作符的特殊对象。这和C/C++函数指针是有本质区别的,因为C/C++的函数指针变量并不具有对象性质,它只是单纯的函数入口地址。上面的Fn只能指向Add和Multiple两个普通函数,无法指向Adder类的Add方法。因为Adder类的Add方法的签名并非int(*)(int a, int b),编译器会自动加上一个隐式的this指针参数,所以它的签名是类似int(*)(Adder *const this, int a, int b) 的。

如果需要指向成员函数的指针,需要用typedef int(Adder::* Fm)(int a, int b)这样的形式加上类型限定符。所以,C++函数指针不能像C#委托一样指向不同类的方法;不具有对象的状态性质;在使用上函数指针也不如委托灵活。所以,当听到“委托就是类似C/C++函数指针”的说法的时候应该既理解其相似之处,又明了其差别。
 

【编辑推荐】

  1. C++虚函数示例解读
  2. C++ Traits应用技巧分享
  3. C++静态构造函数相关概念解析
  4. C++框架如何正确理解
  5. C++跨平台应用编译方法详解

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
Windows恶意软件20年“... 在Windows的早期年代,病毒游走于系统之间,偶尔删除文件(但被删除的文件几乎都是可恢复的),并弹...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
着眼MAC地址,解救无法享受D... 在安装了DHCP服务器的局域网环境中,每一台工作站在上网之前,都要先从DHCP服务器那里享受到地址动...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...