当前位置: 首页 > news >正文

【C++】什么是函数对象和函数对象的用处

目录

什么是函数对象

函数对象的用处


什么是函数对象

class myprint
{
    public:
    void operator()(int num)
    {
        cout << "num " << num << endl;
        count++; 
    }
    int count = 0;
};

void text2(int num)
{
    cout << "num " << num << endl;
}

void test1()
{
    //myprint 是一个类 而不是函数
    myprint text;

    text(11); //仿函数调用
    text2(11); //函数调用

    myprint()(100);
}

//函数对象超出普通函数的概念, 内部可以保存状态
void test2()
{
    myprint q;

    q(10);
    q(20);
    q(30);
    q(40);
    q(50);

    cout << q.count  << endl;
}

//函数对象做参数
void doprint(myprint print, int num)
{
    print(num);
}

void test3()
{
    doprint(myprint(), 30);
}

转自:https://www.jianshu.com/p/1c986b510cff 

函数对象的用处

c++中函数对象是一种类似于函数指针的C++函数对象,C++函数对象不是函数指针。但是,在程序代码中,它的调用方式与函数指针一样,后面加个括号就可以了。

C++函数对象实质上是一个实现了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

呵呵,除了定义方式不一样,使用方式可是一样的。都是:

  
  1. cout << add(3,2); 

既然C++函数对象与函数指针在使用方式上没什么区别,那为什么要用函数对象呢?很简单,函数对象可以携带附加数据,而指针就不行了。下面就举个使用附加数据的例子:

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 的位置

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

要想让一个函数既能接受函数指针,也能接受函数对象,最方便的方法就是用模板。如:

  
template<typename FUNC> 
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

另外,C++函数对象还有一个函数指针无法匹敌的用法:可以用来封装类成员函数指针!因为函数对象可以携带附加数据,而成员函数指针缺少一个类实体(类实例)指针来调用,因此,可以把类实体指针给函数对象保存起来,就可以用于调用对应类实体成员函数了。

template<typename O> 
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<A> 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已经被接受的一个提案,就是可变模板参数列表。用了这个技术,就不需要偏特化无数个C++函数对象了,只要一个函数对象模板就可以解决问题了。

https://www.cnblogs.com/liqilei/articles/1865715.html

相关文章:

  • 【C++】STL标准容器的排序操作和选择合适的排序算法
  • 【C++】程序猿c++(11) 字符串比较误区总结
  • 【C++】C++ STL中 next_permutation,prev_permutation函数的用法
  • 【C++】search、search_n和find、find_if
  • 【C++】迭代器iterator研究(input iterator、output iterator等)----编辑中
  • 【WebRTC】WebRTC介绍及简单应用
  • 【高并发】高并发测试笔记
  • 【SSL】HTTPS 和 SSL证书原理
  • 【mySQL】WAL和MVCC ----待消化
  • 【GDB】GDB 调试多线程和多进程总结
  • 【C++】C++的类|C++类的内存结构
  • 【算法】详解二分查找算法(思路很简单,细节是魔鬼)
  • 【数据结构】linux 内核的list
  • 【linux】信号量与PV操作 (进程和线程的同步)
  • 【内存池】C++ 内存池
  • 【EOS】Cleos基础
  • Apache Zeppelin在Apache Trafodion上的可视化
  • IDEA常用插件整理
  • JS变量作用域
  • magento 货币换算
  • Median of Two Sorted Arrays
  • ng6--错误信息小结(持续更新)
  • PHP 的 SAPI 是个什么东西
  • Python socket服务器端、客户端传送信息
  • rabbitmq延迟消息示例
  • React as a UI Runtime(五、列表)
  • vue-cli在webpack的配置文件探究
  • 从@property说起(二)当我们写下@property (nonatomic, weak) id obj时,我们究竟写了什么...
  • 从零到一:用Phaser.js写意地开发小游戏(Chapter 3 - 加载游戏资源)
  • 第13期 DApp 榜单 :来,吃我这波安利
  • 构造函数(constructor)与原型链(prototype)关系
  • 前端技术周刊 2018-12-10:前端自动化测试
  • 前端技术周刊 2019-01-14:客户端存储
  • 实战|智能家居行业移动应用性能分析
  • 使用 Node.js 的 nodemailer 模块发送邮件(支持 QQ、163 等、支持附件)
  • 新版博客前端前瞻
  • 原生 js 实现移动端 Touch 滑动反弹
  • 职业生涯 一个六年开发经验的女程序员的心声。
  • 好程序员大数据教程Hadoop全分布安装(非HA)
  • ​Distil-Whisper:比Whisper快6倍,体积小50%的语音识别模型
  • ​一文看懂数据清洗:缺失值、异常值和重复值的处理
  • # Swust 12th acm 邀请赛# [ E ] 01 String [题解]
  • (17)Hive ——MR任务的map与reduce个数由什么决定?
  • (二)pulsar安装在独立的docker中,python测试
  • (附源码)springboot 房产中介系统 毕业设计 312341
  • (附源码)springboot建达集团公司平台 毕业设计 141538
  • (附源码)ssm跨平台教学系统 毕业设计 280843
  • (解决办法)ASP.NET导出Excel,打开时提示“您尝试打开文件'XXX.xls'的格式与文件扩展名指定文件不一致
  • (顺序)容器的好伴侣 --- 容器适配器
  • (四)【Jmeter】 JMeter的界面布局与组件概述
  • .net core 6 使用注解自动注入实例,无需构造注入 autowrite4net
  • @GlobalLock注解作用与原理解析
  • @SpringBootApplication 包含的三个注解及其含义
  • []FET-430SIM508 研究日志 11.3.31
  • [2024] 十大免费电脑数据恢复软件——轻松恢复电脑上已删除文件