14-8 C++函数调用运算符
14.8.0 引入
函数对象
我们可以重载类类型的函数调用运算符,进而像使用函数一样使用这个类
那么这个类既可以存储状态,又可以当函数使用,十分灵活
//定义一个类,让它起到一个求绝对值函数的作用
struct absInt{
int operator()(int val) const {
return val<0? -val : val;
}
};
//使用
int main(){
int i = -42;
absInt absObj;
int ui = absObj(i);
cout<<ui<<endl;
return 0;
}
函数调用运算符必须是成员函数。一个类可以定义多个不同版本的调用运算符[这样就像函数的重载],相互之间应该在参数数量或类型上有所区别。
如果类定义了调用运算符,则该类的对象称作函数对象( function object)。因为可以调用这种对象,所以我们说这些对象的“行为像函数一样”。
含有状态的函数对象
举个例子,我们将定义一个打印 string 实参内容的类。默认情况下,我们的类会将内容写入到cout中,每个string之间以空格隔开。同时也允许类的用户提供其他可写入其他分隔符。我们将该类定义如下
class PrintString{
public:
PrintString(ostream &o = cout, char c = ' ') :
os(o), sep(c){};
void operator()(const string &s) const{
os<<s<<sep;
}
private:
ostream &os;
char sep;
};
//上述的类等价于下面的函数
void PrintString_Function(const string &s,
ostream &os = cout, char sep = ' '){
os<<s<<sep;
}
int main(){
string s("Hello World!");
PrintString ps;
ps(s);
return 0;
}
函数对象常做泛型算法实参
和lambda一样,函数对象常做泛型算法的实参
如for_each
for_each(vs.begin(), vs.end(), PrintString(cout,'\n'));
//等价于
for_each(vs.begin(), vs.end(),
[](const string &s){cout<<s<<endl;});
14.8.1 lambda是函数对象
表示没有捕获值的lambda的类
在第十章泛型算法的“10-3定制操作lambda”中我们介绍了lambda的使用,但其实lambda本质上就是重载了()的未命名类的未命名对象,如10-3的例子
Stable_sort(words.begin(), words.end(),
[](const string &a, const string &b)
{return a.size() < b.size();});
//lambda的行为类似于下面的类的一个未命名对象
class ShorterString{
bool operator()(const string &a, const string &b)
{ return a.size()<b.size();}
};
//Stable_sort还可以写为
Stable_sort(words.begin(), words.end(), ShorterString());
表示lambda及其捕获行为的类
众所周知,lambda有两种捕获行为:引用捕获和值捕获
- 引用捕获时:程序负责确保执行时引用的对象确实存在,所以可以直接使用该对象的引用,而不必在lambda产生的类中存储为数据成员
- 值捕获时:lambda必须将每个值捕获的对象保存为数据成员
//获得第一个指向满足条件元素的迭代器,该元素满足size()>=sz
auto wc = find_if(words.begin(), words.end(),
[sz](const string &s)
{return s.size() >= sz;});
//lambda等价于下面的类的未命名对象
class SizeComp{
public :
SizeComp(int n) : sz(n){}
//该调用类型的形参,函数体和返回类型都和lambda一致
bool operator()(const string s){
return s.size() >= sz;
}
private:
int sz;
};
//等价调用
auto wc = find_if(words.begin(), words.end(),
SizeComp(sz));
14.8.2 标准库定义的函数对象
标准库定义了一组表示算术运算符、关系运算符和逻辑运算符的类
例如:
- plus类定义了一个函数调用运算符用于对一对运算对象执行+的操作;
- modulus类定义了一个调用运算符执行二元的%操作;
- equal_to类执行==
这些类都被定义成模板的形式
例如:
- plus<string>令string加法运算符作用于string对象;
- plus<int>的运算对象是int;
- plus<sales_data>对 sales_data对象执行加法运算

在算法中使用高标准库函数

标准库规定其函数对象对于指针同样适用。
我们之前曾经介绍过比较两个无关指针将产生未定义的行为(参见3.5.3节,第107页),然而我们可能会希望通过比较指针的内存地址来sort指针的vector。
直接这么做将产生未定义的行为,因此我们可以使用一个标准库函数对象来实现该目的:

14.8.3 可调用对象与function
调用形式
C语言中可调用对象
- 函数和函数指针
- lambda
- 重载了函数调用运算符的类
不同的调用对象可能有相同的调用形式,如
//普通函数
int add(int i, int j) { return i+j;}
//lambda,产生一个未命名的函数对象类
auto mod = [](int i, int j){ return i%j;};
//函数对象类
struct divide{
int operator()(int denominator, int divisor){
return denominator/divisor;
}
};
上述三种对象都享有相同的调用形式
int ( int, int)
用function统一调用形式
function定义在头文件functional中

function<int(int, int)> f1 = add; //函数指针
function<int(int, int)> f2 = divide(); //函数对象类的对象
function<int(int, int)> f3 = [](int i, int j)//lambda
{return i*j;};
cout<<f1(4,2)<<endl; //打印6
cout<<f2(4,2)<<endl; //打印2
cout<<f3(4,2)<<endl; //打印8
用function建立函数表
函数表(function table)中不知包含函数,而是包含各种可调用对象(函数,lambda,函数对象类的对象)
//列举了可调用对象与二元运算符对应关系的表格
//所有可调用对象都必须接受两个int、返回一个int
//其中的元素可以是函数指针、函数对象或者lambda
map<string ,function<int(int, int)> binops = {
{"+", add}, //函数指针
{"-", minus<int>()}, //标准库函数对象
{"/", divide()}, //用户定义的函数对象
{"*",[](int i, int j){return i*j;}}, //未命名lambda
{"%", mod} //命名了的lambda对象
};
//使用:
binops["+"](10,5); //调用add(10,5)
binops["-"](10,5); //使用minus<int>对象的调用运算符
binops["/"](10,5); //使用divide对象的调用运算符
binops["*"](10,5); //调用lambda函数对象
binops["%"](10,5); //调用lambda函数对象
14-8 C++函数调用运算符的更多相关文章
- C++ Pirmer : 第十四章 : 重载运算符与类型转换之函数调用运算符与标准库的定义的函数对象
函数调用运算符 struct test { int operator()(int val) const { return (i > 0 ? i : -i); } }; 所谓的函数调用就是一个类重 ...
- 函数调用运算符"()"
14.8函数调用运算符"()"1.函数调用运算符必须是成员函数,一个类可以定义多个不同版本的调用运算符,但是他们相互之间应该在参数数量或返回类型上有所区别.定义了调用运算符的类的对 ...
- C++ //函数调用运算符重载 (仿函数)
1 //函数调用运算符重载 2 3 #include <iostream> 4 #include <string> 5 using namespace std; 6 7 //函 ...
- C++赋值运算符、函数调用运算符、下标运算符(“=”、“()”、“[]”)重载
#include <iostream>#include <assert.h>#include <string.h> using namespace std; cla ...
- c/c++ 重载运算符 函数调用运算符
重载运算符 函数调用运算符 把一个类的对象a,当成函数来使用,比如a(),所以需要重载operator()方法.重载了函数调用运算符的类的对象,就是函数对象了. 还有什么是函数对象呢??? lambd ...
- [C++] 重载运算符与类型转换(2)——函数调用运算符和类型转换运算符
1.这两个应该是C++中比较高级的用法了. 一.函数调用运算符 1.重载函数调用运算符(),必须是成员函数,一个类可以定义多个不同版本的调用运算符,相互之间应该在参数数量或者类型上有所区别. ...
- [C++ Primer] : 第14章: 重载运算符与类型转换
基本概念 重载运算符是具有特殊名字的函数: 它们的名字由关键字operator和其后要定义的运算符号共同组成. 重载运算符函数的参数数量与该运算符作用的运算对象数量一样多. 对于二元运算符来说, 左侧 ...
- C++学习之路—运算符重载(二)运算符重载作为类的成员函数和友元函数
(根据<C++程序设计>(谭浩强)整理,整理者:华科小涛,@http://www.cnblogs.com/hust-ghtao转载请注明) 对运算符重载的函数有两种处理方式:(1)把运算符 ...
- C++ Primer 5th 第14章 重载运算与类型转换
当运算符作用域类类型的对象时,可以通过运算符重载来重新定义该运算符的含义.重载运算符的意义在于我们和用户能够更简洁的书写和更方便的使用代码. 基本概念 重载的运算符是具有特殊名字的函数:函数名由关键词 ...
- [转]C++之运算符重载(1)
在前一节中曾提到过,C++中运行时的多态性主要是通过虚函数来实现的,而编译时的多态性是由函数重载和运算符重载来实现的.这一系列我将主要讲解C++中有关运算符重载方面的内容.在每一个系列讲解之前,都会有 ...
随机推荐
- Kafka Topic 中明明有可拉取的消息,为什么 poll 不到
开心一刻 今天小学女同学给我发消息她:你现在是毕业了吗我:嗯,今年刚毕业她给我发了一张照片,怀里抱着一只大橘猫她:我的眯眯长这么大了,好看吗我:你把猫挪开点,它挡住了,我看不到她:你是 sb 吗,滚我 ...
- SimpleTranslationAIAgent:基于C#与LLM的翻译AI Agent
基于C#与LLM通过简单对话即可实现文件到文件的翻译任务 该软件是MIT协议完全开源免费的,但是调用LLM的API可能需要费用,但是没关系,赛博菩萨硅基流动与智谱AI等都有免费的模型可调了. 这个Tr ...
- 根据Uri,Cursor没有获取到对应的属性
Android: 背景:调用摄像头,拍摄视频,指定保存的地址,但是返回的Cursor文件,只有名称和大小的属性,没有其他诸如时长,连ID属性都没有 使用 cursor.getInt(cursor.ge ...
- 已知两个长度分别为m和n的升序链表,若将它们合并为长度为m+n的一个降序链表,则最坏情况下的时间复杂度是
已知两个长度分别为m和n的升序链表,若将它们合并为长度为m+n的一个降序链表,则最坏情况下的时间复杂度是(). 解析:选D 两个升序合并为降序,操作就不多说了,两数列依次比较放入,其中一个数列结束了, ...
- vscode新建html文件并快速生成标准的html代码
1.打开vscode,点击Open Folder 2.选择目标文件夹,新建一个扩展名为.html的文件: 3.在第一行输入 !(英文状态下),按tab键,新建成功.界面如下图所示:
- Google sheet
最近做比较多 data migration 的东西. 当我们开发一个新的系统去替代一个旧系统时,通常就需要做大量的 migration 动作. 有好几个做法 我之前比较常用的的工具是 sql 和 c# ...
- Asp.net Core 学习笔记 Azure Storage
更新: 2021-07-22 使用 Azure storage 以后, 还要解决一个 url 的问题. 文件自然是通过我们的 domain 来访问才合理丫. 这个是 azure 的 url : htt ...
- 使用 Wake Lock API:保持设备唤醒的最佳实践
在现代 Web 应用中,尤其是涉及视频播放.实时通信.地图导航等长时间运行的任务时,用户常常希望设备不要因为空闲而自动进入睡眠模式或屏幕变暗.为了解决这一问题,Web API 提供了一个名为 Wake ...
- 还在苦于密码太弱?教你3招用Linux生成高强度密码
各位好啊,我是会编程的蜗牛,作为java开发者,我们平常肯定会接触Linux操作系统,其实除了一般的部署应用外,它还可以帮助我们生成密码.解决我们平常自己想各种复杂密码的烦恼,以后我会讲一讲如何安全地 ...
- QQ或者微信可以放昵称的超好看的符号
☪︎⋆ ✯ ⛈ •ᴗ• •ᴥ• ◔.̮◔ ᕱ ᕱ ⸝⸝· ᴥ ·⸝⸝ ʕ·͡ˑ·ཻʔ ʕ•̫͡•ོʔ ˃̣̣̥᷄⌓˂̣̣̥᷅ °꒰'ꀾ'꒱° ⋆ᶿ̵᷄ ˒̼ ᶿ̵᷅⋆ ˙ϖ˙ ⚝ ︎ .˗ˏˋ♡ˎˊ˗ ...