转载自 http://www.cnblogs.com/weidagang2046/archive/2009/08/09/1542248.html 感谢作者分享!

我们常见到C#技术文献用“类似C/C++函数指针的东西”介绍委托。这样好像是有道理的,因为二者的确有深层次的相通之处。委托和函数指针都描述了方法/函数的签名,并通过统一的接口调用不同的实现。但二者又有明显的区别,简单说来,委托对象是真正的对象,而函数指针变量只是函数的入口地址。对于高层应用来讲,委托的灵活性和适用范围更胜函数指针;但对于底层应用,函数指针则是不可替代的。下面分别是委托类型和函数指针类型定义的例子:

delegate int Fn(int a, int b) //C#委托

typedef int (*Fn)(int a, int b) //C++函数指针

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

//C#

delegate int Fn(int a, int b) ;

class Adder{

private int c = 0;

public int Add(int a, int b){

return a + b + c;

}

public Adder(int c){ this.c = c; }

}

class Multiplier{

private int c = 0;

public int Multiple(int a, int b){

return a * b * c;

}

public Multiplier(int c){ this.c = c; }

}

Adder adder = new Adder(1);

Multiplier multiplier = new Multiplier(2);

Fn fn = adder.Add;

fn(1, 2); //结果为4

fn = multiplier.Multiple;

fn(2, 3); //结果为12

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

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

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

//C++

typedef int(*Fn)(int a, int b);

int Add(int a, int b) { 
    return a + b; 
};

int Multiple(int a, int b) { 
    return a * b; 
};

class Adder { 
public: 
    Adder(int c) { 
        this->c = c; 
    } 
    int Add(int a, int b) { 
        return a + b + c;  
    } 
private: 
    int c; 
};

typedef int(Adder::* Fm)(int a, int b);

int _tmain(int argc, _TCHAR* argv[]) 

    Fn fn = Add; 
    std::cout << fn(1, 2) << std::endl;

fn = Multiple; 
    std::cout << fn(1, 2) << std::endl;

Adder adder(1); 
    Fm f = &Adder::Add; 
    std::cout << (adder.*f)(1, 2) << std::endl; 
    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++函数指针”的说法的时候应该既理解其相似之处,又明了其差别。

Functor

我们常说C++是强大而复杂的语言,函数指针已经被C#委托PK下来了,难道C++就没有可以PK C#委托的大将吗?当然有!首先应该看到,函数指针并非C++的产物,而是继承自C,因此函数指针的局限其实是C的局限,与C++无关。我们说C#委托是支持()函数调用操作符的特殊对象,在C++中()操作符是可以被重载的,我们把重载了()操作符的对象称为functor。下面是一个functor的例子:

class Adder { 
public: 
    Adder(int c) { 
        this->c = c; 
    } 
    int operator()(int a, int b) { 
        return a + b + c; 
    } 
private: 
    int c; 
};

int _tmain(int argc, _TCHAR* argv[]) {

Adder adder(1); 
    std::cout << adder(2, 3) << std::endl; //输出6, adder是functor-重载了()操作符的对象

return 0; 
}

functor和委托对象有相似之处,都是支持()操作符的具有状态的对象。但functor还不是委托,为什么?因为委托类型是一种接口规范,函数指针类型也是,但functor本身不是接口规范。这里,请注意区分类型和对象之间的关系:委托类型和委托对象,函数指针类型和函数指针变量。functor可以等同于委托对象,但如何表达委托类型呢?应该看到,委托机制是一种运行时的动多态,委托对象可以与目标方法动态地绑定。由于C++并非基于虚拟机的语言,想直接动态绑定不同类型functor到统一的类型接口并不容易。但C++有一种强大的工具实现静多态,这就是模版:

模版

class Adder { 
public: 
    Adder(int c) { 
        this->c = c; 
    } 
    int operator()(int a, int b) { 
        return a + b + c; 
    } 
private: 
    int c; 
};

class Multiplier { 
public: 
    Multiplier(int c) { 
        this->c = c; 
    }

int operator()(int a, int b) { 
        return a * b * c; 
    } 
private: 
    int c; 
};

template<typename T> 
void Do(T& f, int a, int b) {

int r = f(a, b); 
    std::cout << r << std::endl; 
};

int _tmain(int argc, _TCHAR* argv[]) { 
    Adder adder(1); 
    Do(adder, 1, 2); //输出4

Multiplier multiplier(10); 
    Do(multiplier, 2, 3);  //输出60

return 0; 
}

可以认为函数模版Do表达了接口规范,它要求一个泛型类T支持()运算符、接受两个整形参数、返回可隐式转换为int的类型。对于不同类型Adder和Multiplier,编译器在编译时自动根据模版为不同的类型T生成了不同的Do重载函数,因此Do的调用形式是统一的,这就是所谓的静多态。有人谈到“模版为静态类型的C++赋予了几乎无类型的元编程能力”,这个例子算是个小小的窥视。STL中许多库方法都是通过模版来表达接口规范,调用者传入functor,其灵活性完全不输C#委托。

总结

1.C#委托对象是真正的对象,C/C++函数指针只是函数入口地址

2.C++的委托对象:functor

3.C++的静多态:模版

转载于:https://www.cnblogs.com/iacocca/archive/2011/10/17/2215789.html

(转载)委托与函数指针辨析相关推荐

  1. C#未来新特性:静态委托和函数指针

    C#每发布一次新版本,都会增加一些底层相关的新特性, 这些特性对大多数商业程序来说并没什么实际用处,主要用于那些对性能有很高要求的代码,如图形处理.机器学习以及数学工具包,等等. 接下来的两个提案,主 ...

  2. C# 未来新特性:静态委托和函数指针

    C# 每发布一次新版本,都会增加一些底层相关的新特性, 这些特性对大多数商业程序来说并没什么实际用处,主要用于那些对性能有很高要求的代码,如图形处理.机器学习以及数学工具包,等等. 接下来的两个提案, ...

  3. C#委托与C++函数指针的区别

    委托与函数指针辨析 我们常见到C#技术文献用"类似C/C++函数指针的东西"介绍委托.这样好像是有道理的,因为二者的确有深层次的相通之处.委托和函数指针都描述了方法/函数的签名,并 ...

  4. 木老师教笨笨课堂——系列讲座(从函数指针到委托) 四、C#的委托

    四.C#的委托 ".Net以委托的形式实现了函数指针的概念."--<C#高级编程(第四版)> 现在看这句话,可能笨笨同学就有感觉了. 看书可能就是这样,想当初天山童姥的 ...

  5. 成员函数指针与高性能的C++委托(三)

    委托(delegate) 和成员函数指针不同,你不难发现委托的用处.最重要的,使用委托可以很容易地实现一个Subject/Observer设计模式的改进版[GoF, p. 293].Observer( ...

  6. 2017/3/8 函数指针/事件/委托....

    函数指针: 定义:函数指针是指向函数的指针变量. 因而"函数指针"本身首先应是指针变量,只不过该指针变量指向函数.这正如用指针变量可指向整型变量.字符型.数组一样,这里是指向函数. ...

  7. 【转载】恼人的函数指针(一)

    本文转载自: http://www.cnblogs.com/AnnieKim/archive/2011/11/20/2255813.html#undefined> 这篇是为了加深记忆所写.发现, ...

  8. 成员函数指针与高性能的C++委托

    成员函数指针与高性能的C++委托 http://www.cnblogs.com/jans2002/archive/2006/10/13/528160.html Member Function Poin ...

  9. 成员函数指针与高性能的C++委托(上篇)

    成员函数指针与高性能的C++委托(上篇) 撰文:Don Clugston 引子 标准C++中没有真正的面向对象的函数指针.这一点对C++来说是不幸的,因为面向对象的指针(也叫做"闭包(clo ...

最新文章

  1. 图灵奖得主LeCun力推无监督学习:要重视基于能量的学习方法
  2. Django博客系统(404页面展示)
  3. 读写分离原来这么简单,一个小注解就够了
  4. 【Qt】Qt 开发桌面程序 ( Qt 版本 5.14.2 | 编辑 Qt 桌面按钮控件 | 修改按钮文本 | 为按钮添加点击事件 | 系统调用 | 去掉系统调用命令窗口 )
  5. JMeter基础知识
  6. LeetCode 2090. 半径为 k 的子数组平均值(滑窗)
  7. C++关联容器,STL关联容器
  8. python构造函数_Python构造函数
  9. Linux系统升级维护费,Antergos Linux 宣布停止维护
  10. 帆软报表使用心得(转)
  11. 浅谈北美油气行业1--加拿大
  12. 路由模式和桥接模式的区别
  13. matlab:pause命令
  14. Tableau豆瓣电影数据项目实战练习1
  15. Python解二元一次方程
  16. LEP(Linux Easy Profiling)2017年度颁奖典礼在西邮隆重举行
  17. 201506051031_《JavaScript权威指南》(p104-143)
  18. redis实现简单的动态密码
  19. linux下测试ftp传输,linux下ftp命令使用详解---linux文件传输ftp命令
  20. No.1 STM32F429IGT6开发板简介 (STM32F429/F767/H743)

热门文章

  1. Ubuntu14.04 indigo ROS的一些使用指令
  2. Redis总结之发布订阅
  3. sublime编辑器无法正常打印中文问题解决
  4. 由Object.prototype.toString.call( )引发关于toString( )方法的思考
  5. Mybatis系列(四):Mybatis缓存
  6. jquery代码小片段
  7. postgresql 字符串拼接||“的使用
  8. css实现强制不换行/自动换行/强制换行
  9. Oracle中添加自动编号的序列
  10. Js中Reflect对象