C++ Pirmer : 第十四章 : 重载运算符与类型转换之函数调用运算符与标准库的定义的函数对象
函数调用运算符
struct test {
int operator()(int val) const { return (i > 0 ? i : -i); }
};
所谓的函数调用就是一个类重载了函数调用符,类在使用重载函数调用符时接受相应参数。这一过程就像是使用一个函数一样,因此叫做函数调用。
int main(void)
{
test t;
int val = -3;
std::cout << "abs(val) = " << t(val) << std::endl;
return 0;
}
使用函数重载调用符时,直接在类名后面跟括号,括号里加所需参数即可,看起来就和一个普通的函数调用一样。 我们称这样的类为“行为像函数一样的类”。
class printString {
public:
printString(std::ostream& o = std::cout) : os(o) {}
void operator()(const std::string& s) { os << "s = " << s << std::endl; }
private:
std::ostream& os;
};
int main(void)
{
std::vector<std::string> vec = { "Hello", "World" };
printString ps;
for_each(vec.begin(), vec.end(), ps);
return 0;
}
lambda是函数对象
stable_sort(words.begin(), words.end(), [](const std::string& a, const std::string& b) { return a.size() < b.size(); });
class Shorter {
public:
bool operator()(const std::string& a, const std::string& b) const { return a.size() < b.size(); }
};
因为默认的lambda是不可以修改捕获参数的,因此上面这个生成的类似的类的函数调用运算符是一个const成员函数。
auto wc = find_if(words.begin(), words.end(), [sz](const std::string& a) { return a.size() >= sz; });
该lambda产生的类形如:
class SizeComp {
public:
SizeComp(size_t n) : sz(n) {}
bool operator()(const std::string& a) const { return a.size() >= sz; }
private:
size_t sz
};
lambda产生的类不含默认构造函数、赋值运算符及默认析构函数,它是否有默认的拷贝/移动构造函数则视捕获的数据成员的类型决定。
习题14.38
class CheckString {
public:
bool operator()(const std::string& s) { return (s.size() >= 1 && s.size() <= 10); }
size_t operator()(std::vector<std::string>& vec, size_t length) {
size_t num = 0;
for (const auto& elmt : vec)
(elmt.size() == length) ? ++num : num;
return num;
}
};
int main()
{
CheckString ck;
std::string s{"HelloWorld"};
std::vector<std::string> vec = { "hello", "world", "i", "love", "you", "but", "you", "is", "what" };
std::cout << "s.size() >= 1 && s.size() <= 10 ?" << ck(s) << std::endl;
size_t n = 1;
std::cout << "size == 1 has " << ck(vec, n) << std::endl;
return 0;
}
标准库定义的函数对象
算术:
plus<Type> // 加法
minus<Type> // 减法
multiplies<Type> // 乘法
divides<Type> // 除法
modulus<Type> // 取模
negate<Type> // 取反 关系:
equal_to<Type> // 等于
not_equal_to<Type> // 不等于
greater<Type> // 大于
greater_equal<Type> // 大于等于
less<Type> // 小于
less_equal<Type> // 小于等于 逻辑:
logical_and<Type> // 逻辑与
logical_or<Type> // 逻辑或
logical_not<Type> // 逻辑非
std::vector<std::string*> vec;
// 未定义行为!
sort(vec.begin(), vec.end(), [](std::string* a, std::string* b){ return a < b; }); sort(vec.begin(), vec.end(), less<std::string*>());
关联容器使用less<key_type>对元素排序。
functional<T> f; f是一个用来存储可调用对象的空function,这些可调用对象的调用形式应该与函数类型T相同
functional<T> f(nullptr); 显示的构造一个空function
functional<T> f(obj); 在f中存储可调用对象obj的副本
f 将f做为条件,当f含有一个可调用对象时为真,否则为假
f(args) 调用f中的对象,参数是args 定义为function<T>的成员的类型
result_type 该function类型的可调用对象返回的类型
argument_type 当T有一个或两个实参时定义的类型,当T有一个实参时,argument_type是该类型的同义词
first_argument_type 当有两个实参时,first_argument_type和second_argument_type分别代表两个实参类型
second_argument_type
function是一个模板, 创建一个function时必须指定它的类型。
int add(int i, int j) { return i / j; }
auto mod = [](int x, int y){ return x % y; };
struct divide {
int operator()(int a, int b) { return a / b; }
};
std::function<int(int, int)> f1 = add;
std::function<int(int, int)> f2 = divide();
std::function<int(int, int)> f3 = [](int x, int y){ return x % y; };
std::cout << f1(4, 2) << std::endl;
std::cout << f2(4, 2) << std::endl;
std::cout << f3(4, 2) << std::endl;
int add(int i, int j) { return i / j; }
Sales_data add(const Sales_data&, const Sales_data&);
int add(int i, int j) { return i / j; }
Sales_data add(const Sales_data&, const Sales_data&);
std:map<std::string, std::function<int(int, int)>> mp;
mp.insert({"+", add}); // 错误,哪个add?
可以使用函数指针来避免二义性的问题:
int (*fp)(int, int) = add;
mp.insert({"+", fp});
或者还可以使用lambda来消除二义性。
C++ Pirmer : 第十四章 : 重载运算符与类型转换之函数调用运算符与标准库的定义的函数对象的更多相关文章
- C++ Primer : 第十四章 : 重载运算与类型转换之重载运算符
重载前须知 重载运算符是特殊的函数,它们的名字由operator和其后要重载的运算符号共同组成. 因为重载运算符时函数, 因此它包含返回值.参数列表和函数体. 对于重载运算符是成员函数时, 它的第一个 ...
- C++ Primer : : 第十四章 : 重载运算符与类型转换之类型转换运算符和重载匹配
类型转换运算符 class SmallInt { public: SmallInt(int i = 0) : val(i) { if (i < 0 || i > 255) throw st ...
- 【C++】《C++ Primer 》第十四章
第十四章 重载运算与类型转换 一.基本概念 重载运算符是具有特殊名字的函数:由关键字operator和其后要定义的运算符号共同组成.也包含返回类型.参数列表以及函数体. 当一个重载的运算符是成员函数时 ...
- 《OpenCL异构并行编程实战》第十二至十四章
▶ 第十二章,在其他语言中使用 OpenCL ● JOCL(Java Building for OpenCL),PyOpenCL ● 一个 PyOpenCL 的例子代码,需要 pyopencl 包 i ...
- 《Linux命令行与shell脚本编程大全》 第十四章 学习笔记
第十四章:呈现数据 理解输入与输出 标准文件描述符 文件描述符 缩写 描述 0 STDIN 标准输入 1 STDOUT 标准输出 2 STDERR 标准错误 1.STDIN 代表标准输入.对于终端界面 ...
- perl 第十四章 Perl5的包和模块
第十四章 Perl5的包和模块 by flamephoenix 一.require函数 1.require函数和子程序库 2.用require指定Perl版本二.包 1.包的定义 2.在包间切 ...
- Gradle 1.12 翻译——第十四章. 教程 - 杂七杂八
有关其它已翻译的章节请关注Github上的项目:https://github.com/msdx/gradledoc/tree/1.12,或訪问:http://gradledoc.qiniudn.com ...
- C和指针 (pointers on C)——第十四章:预处理器
第十四章 预处理器 我跳过了先进的指针主题的章节. 太多的技巧,太学科不适合今天的我.但我真的读,读懂.假设谁读了私下能够交流一下.有的小技巧还是非常有意思. 预处理器这一章的内容.大家肯定都用过.什 ...
- CSS3秘笈复习:十三章&十四章&十五章&十六章&十七章
第十三章 1.在使用浮动时,源代码的顺序非常重要.浮动元素的HTML必须处在要包围它的元素的HTML之前. 2.清楚浮动: (1).在外围div的底部添加一个清除元素:clear属性可以防止元素包围浮 ...
随机推荐
- 第四章· ucos系统及其任务
来自为知笔记(Wiz)
- Scrum Meeting 8-20151210
任务安排 姓名 今日任务 明日任务 困难 董元财 服务器入口更新 服务器发布记录接口 无 胡亚坤 请假(编译攻坚) 聊天界面优化 无 刘猛 请假(编译攻坚) 开新的分支存放服务器端或者,创建的仓库记录 ...
- Vim入门教程
尽管网上有成打的Vim在线教程,但是要么艰深晦涩,要么太过肤浅.本教程的目标让每个阶段都有斩获,从理解它的哲学(将和你终身相伴)到超越现在编辑技巧,成为其中的牛人. 简单来说,本教程的学习方式将使你终 ...
- GridFS图片
-----------2016-5-9 18:58:56-- source:GridFS实现图片的存取
- 基于gulp 的前端自动化构建方案总结
一,基础篇 先安装nodejs 使用淘宝镜像安装tnpm 安装 cnpm 插件:npm install -g cnpm --registry=https://registry.npm.taobao.o ...
- Maven3简介
Maven3简介 百度百科 maven官网: http:\/\/maven.apache.org\/ Maven下载安装 Maven 下载:http:\/\/maven.apache.org\/dow ...
- iOS开发 Apple Pay
一.什么是Apple Pay? 1. 概念 Apple Pay,简单来说, 就是一种移动支付方式.通过Touch ID/ Passcode,用户可使用存储在iPhone 6, 6p等设备上的信用卡和借 ...
- js中对类和对象的理解
类 :对一群具有相同特征的对象的集合的描述:对象:真实存在的对象个体: **面向对象,而不是面向类. 1.一切皆对象,继承靠原型链,多态靠弱类型,封装--虽然可以靠闭包,但我个人更推崇和python一 ...
- SPSS数据分析—非线性回归
线性回归的首要满足条件是因变量与自变量之间呈线性关系,之后的拟合算法也是基于此,但是如果碰到因变量与自变量呈非线性关系的话,就需要使用非线性回归进行分析. SPSS中的非线性回归有两个过程可以调用,一 ...
- UNIX 逐字符输入
//终端驱动处于普通胡一次一字符模式 system("stty raw"); //终端驱动处于普通胡一次一行模式 system("stty cooked");