跑男runningman哪一期:函数对象

来源:百度文库 编辑:偶看新闻 时间:2024/04/28 05:53:45
函数对象不是函数指针。但是,在程序代码中,它的调用方式与函数指针一样,后面加个括号就可以了。
  这是入门级的随笔,说的是函数对象的定义,使用,以及与函数指针,成员函数指针的关系。  
沐枫小筑
函数对象实质上是一个实现了operator()--括号操作符--的类。
例如:
class Add
{
public:
  int operator()(int a, int b)
  {
    return a + b;
  }
};
Add add; // 定义函数对象
cout << add(3,2); // 5
函数指针版本就是:
int AddFunc(int a, int b)
{
  return a + b;
}
typedef int (*Add) (int a, int b);
Add add = &AddFunc;
cout << add(3,2); // 5
呵呵,除了定义方式不一样,使用方式可是一样的。都是:
cout << add(3,2);
既然函数对象与函数指针在使用方式上没什么区别,那为什么要用函数对象呢?很简单,函数对象可以携带附加数据,而指针就不行了。
下面就举个使用附加数据的例子:
class less
{
public:
    less(int num):n(num){}
    bool operator()(int value)
    {
        return value < n;
    }
private:
    int n;
};


使用的时候:
    less isLess(10);
    cout << isLess(9) << " " << isLess(12); // 输出 1 0

这个例子好象太儿戏了,换一个:
const int SIZE = 5;
int array[SIZE] = { 50, 30, 9, 7, 20};
// 找到小于数组array中小于10的第一个数的位置
int * pa = std::find_if(array, array + SIZE, less(10)); // pa 指向 9 的位置
// 找到小于数组array中小于40的第一个数的位置
int * pb = std::find_if(array, array + SIZE, less(40)); // pb 指向 30 的位置

这里可以看出函数对象的方便了吧?可以把附加数据保存在函数对象中,是函数对象的优势所在。
它的弱势也很明显,它虽然用起来象函数指针,但毕竟不是真正的函数指针。在使用函数指针的场合中,它就无能为力了。例如,你不能将函数对象传给qsort函数!因为它只接受函数指针。

要想让一个函数既能接受函数指针,也能接受函数对象,最方便的方法就是用模板。如:
template
int count_n(int* array, int size, FUNC func)
{
    int count = 0;
    for(int i = 0; i < size; ++i)
        if(func(array[i]))
            count ++;
    return count;
}
这个函数可以统计数组中符合条件的数据个数,如:
const int SIZE = 5;
int array[SIZE] = { 50, 30, 9, 7, 20};
cout << count_n(array, SIZE, less(10)); // 2
用函数指针也没有问题:
bool less10(int v)
{
    return v < 10;
}
cout << count_n(array, SIZE, less10); // 2
另外,函数对象还有一个函数指针无法匹敌的用法:可以用来封装类成员函数指针!
因为函数对象可以携带附加数据,而成员函数指针缺少一个类实体(类实例)指针来调用,因此,可以把类实体指针给函数对象保存起来,就可以用于调用对应类实体成员函数了。

template
class memfun
{
public:
    memfun(void(O::*f)(const char*), O* o): pFunc(f), pObj(o){}
    void operator()(const char* name)
    {
        (pObj->*pFunc)(name);
    }
private:
    void(O::*pFunc)(const char*);
    O* pObj;
};

class A
{
public:
    void doIt(const char* name)
    { cout << "Hello " << name << "!";}
};

    A a;
    memfun call(&A::doIt, &a); // 保存 a::doIt指针以便调用
    call("Kitty"); // 输出 Hello Kitty!

大功告成了,终于可以方便保存成员函数指针,以备调用了。

不过,现实是残酷的。函数对象虽然能够保有存成员函数指针和调用信息,以备象函数指针一样被调用,但是,它的能力有限,一个函数对象定义,最多只能实现一个指定参数数目的成员函数指针。
标准库的mem_fun就是这样的一个函数对象,但是它只能支持0个和1个参数这两种成员函数指针。如 int A::func()或void A::func(int)、int A::func(double)等等,要想再多一个参数如:int A::func(int, double),不好意思,不支持。想要的话,只有我们自已写了。
而且,就算是我们自已写,能写多少个?5个?10个?还是100个(这也太恐怖了)?
好在boost库提供了boost::function类,它默认支持10个参数,最多能支持50个函数参数(多了,一般来说这够用了。但它的实现就是很恐怖的:用模板部份特化及宏定义,弄了几十个模板参数,偏特化(编译期)了几十个函数对象。

----
C++0x已经被接受的一个提案,就是可变模板参数列表。用了这个技术,就不需要偏特化无数个函数对象了,只要一个函数对象模板就可以解决问题了。期待吧。
求救:用构造函数将n个对象建立成链表 . VC中 有时候对象的成员函数和变量都出不来? 能不能将一个函数返回的结构对象直接赋值给另一结构对象,怎么实现 复制构造函数中调用被应用对象的成员函数 为什么在折构函数中先被创建的对象后被执行,后被创建的对象先被执行?, 构造函数用于创建类的实例对象,构造函数名应与类名相同,返回类型为void. 构造函数里啥内容也没有!!!!为何它能初始化对象呀??? c++中面向对象中的构成函数为什么没有返回类型 [VB]中使用SendMessage函数向对象发送消息的技巧和方法 高手求救:authorware函数\变量\知识对象打开都是空白 authorware7.0 工具栏中的知识对象,变量,函数在使用过程中消失 不同函数中可以使用相同名字的变量,它们代表不同的对象,互不干扰 函数~~~~~~~~~~~~ 函数...... 函数!!!!!!!!!!! 函数!!!!!!!!!!! 函数...... 求一C++程序,要用上类和对象,构造函数,析构函数,继承与派生的知识编一个程序,谢谢! 求一C++程序,要用上类和对象,构造函数,析构函数,继承与派生的知识编一个程序,谢谢!! 操作系统采用面向对象技术,给用户提供的函数所封装的数据结构和程序是怎么实现的? 能在excel中使用3个以上的条件格式吗?用函数怎样使用?对象是文字 编写一个巨型整数类HugeInt,可以用默认值0或长整型数或字符串进行对象的初始化(构造) 然后用成员函数、 c++中是否也像Java中一样,静态成员函数可以在未创建对象时就可以被调用 为什么 我的VC++6.0中 定义某个类的对象时 再用.访问时不自动弹出他的成员函数或者成员变量呢?