C++11新特性应用--实现延时求值(std::function和std::bind)
说是延时求值,注意还是想搞一搞std::function和std::bind。
之前博客《C++11新特性之std::function》注意是std::function怎样实现回调函数。
如今就算是补充吧,再把std::bind进行讨论讨论。
何为Callable Objects?
就可以调用对象,比方函数指针、仿函数、类成员函数指针等都可称为可调用对象。
对象包装器
Function wrapper
Class that can wrap any kind of callable element (such as functions and function objects) into a copyable object, and whose type depends solely on its call signature (and not on the callable element type itself).
An object of a function class instantiation can wrap any of the following kinds of callable objects: a function, a function pointer, a pointer to member, or any kind of function object (i.e., an object whose class defines operator(), including closures).
A decay copy of the wrapped callable object is stored internally by the object, which becomes the function’s target. The specific type of this target callable object is not needed in order to instantiate the function wrapper class; only its call signature.
以下用一段代码:
#include<iostream>
#include<functional>
//普通函数
void func(void)
{
std::cout << "1" << std::endl;
}
//类的成员函数
class A
{
public:
static int A_func(int a)
{
std::cout << "2" << "(" << a << ")" << std::endl;
return a;
}
};
//仿函数
class B
{
public:
int operator()(int a)
{
std::cout << "2" << "(" << a << ")" << std::endl;
return a;
}
};
int main()
{
std::function<void(void)> fun1 = func;
fun1();
std::function<int(int)> fun2 = A::A_func;
std::cout << fun2(123) << std::endl;
B b;
fun2 = b;
std::cout << fun2(123) << std::endl;
return 0;
}
//输出:
1
2(123)
123
2(123)
123
接下来std::function用于回调就不浪费篇幅了,接下来注意分析std::bind。
何为std::bind?
字面意思。绑定器。
simple
template
#include <iostream> // std::cout
#include <functional> // std::bind
// a function: (also works with function object: std::divides<double> my_divide;)
double my_divide(double x, double y) { return x / y; }
struct MyPair {
double a, b;
double multiply() { return a*b; }
};
int main() {
using namespace std::placeholders; // adds visibility of _1, _2, _3,...
// binding functions:
auto fn_five = std::bind(my_divide, 10, 2); // returns 10/2
std::cout << fn_five() << '\n'; // 5
auto fn_half = std::bind(my_divide, _1, 2); // returns x/2
std::cout << fn_half(10) << '\n'; // 5
auto fn_invert = std::bind(my_divide, _2, _1); // returns y/x
std::cout << fn_invert(10, 2) << '\n'; // 0.2
auto fn_rounding = std::bind<int>(my_divide, _1, _2); // returns int(x/y)
std::cout << fn_rounding(10, 3) << '\n'; // 3
MyPair ten_two{ 10,2 };
// binding members:
// returns x.multiply()
auto bound_member_fn = std::bind(&MyPair::multiply, _1);
std::cout << bound_member_fn(ten_two) << '\n'; // 20
// returns ten_two.a
auto bound_member_data = std::bind(&MyPair::a, ten_two);
std::cout << bound_member_data() << '\n'; // 10
return 0;
}
_ 1中的 _ 表示的是占位符,由using namespace std::placeholders; 提供。详细的话找机会再研究。
主要看看上面的代码。bind的几种使用方式。
能够看到。能够绑定所有參数,也能够绑定部分參数。
你可能已经感到bind的威力了吧,那不是重点。与function的结合才是重要的:
//#include <iostream> // std::cout
//#include <functional> // std::bind
//
//// a function: (also works with function object: std::divides<double> my_divide;)
//double my_divide(double x, double y) { return x / y; }
//
//struct MyPair {
// double a, b;
// double multiply() { return a*b; }
//};
//
//int main() {
// using namespace std::placeholders; // adds visibility of _1, _2, _3,...
//
// // binding functions:
// auto fn_five = std::bind(my_divide, 10, 2); // returns 10/2
// std::cout << fn_five() << '\n'; // 5
//
// auto fn_half = std::bind(my_divide, _1, 2); // returns x/2
// std::cout << fn_half(10) << '\n'; // 5
//
// auto fn_invert = std::bind(my_divide, _2, _1); // returns y/x
// std::cout << fn_invert(10, 2) << '\n'; // 0.2
//
// auto fn_rounding = std::bind<int>(my_divide, _1, _2); // returns int(x/y)
// std::cout << fn_rounding(10, 3) << '\n'; // 3
//
// MyPair ten_two{ 10,2 };
//
// // binding members:
// auto bound_member_fn = std::bind(&MyPair::multiply, _1); // returns x.multiply()
// std::cout << bound_member_fn(ten_two) << '\n'; // 20
//
// auto bound_member_data = std::bind(&MyPair::a, ten_two); // returns ten_two.a
// std::cout << bound_member_data() << '\n'; // 10
//
// return 0;
//}
#include<iostream>
#include<functional>
class A {
public:
int i_ = 0;
void output(int x, int y)
{
std::cout << x << " " << y << std::endl;
}
};
int main()
{
A a;
std::function<void(int, int)> func1 = std::bind(&A::output, &a, std::placeholders::_1,
std::placeholders::_2);
func1(1, 2);
std::function<int&(void)> func2 = std::bind(&A::i_, &a);
func2() = 888;
std::cout << a.i_ << std::endl;
return 0;
}
//输出:
1 2
888
C++11新特性应用--实现延时求值(std::function和std::bind)的更多相关文章
- C++11新特性(1) 右值引用
在C++中,左值(lvalue)是能够获取其地址的一个量.因为常常出如今赋值语句的左边.因此称之为左值.比如一个有名称的变量. 比如: int a=10; //a就是一个左值. 传统的C++引用,都是 ...
- [转载] C++11新特性
C++11标准发布已有一段时间了, 维基百科上有对C++11新标准的变化和C++11新特性介绍的文章. 我是一名C++程序员,非常想了解一下C++11. 英文版的维基百科看起来非常费劲,而中文版维基百 ...
- C++ 11学习和掌握 ——《深入理解C++ 11:C++11新特性解析和应用》读书笔记(一)
因为偶然的机会,在图书馆看到<深入理解C++ 11:C++11新特性解析和应用>这本书,大致扫下,受益匪浅,就果断借出来,对于其中的部分内容进行详读并亲自编程测试相关代码,也就有了整理写出 ...
- C++11新特性总结 (二)
1. 范围for语句 C++11 引入了一种更为简单的for语句,这种for语句可以很方便的遍历容器或其他序列的所有元素 vector<int> vec = {1,2,3,4,5,6}; ...
- C++11新特性总结 (一)
1. 概述 最近在看C++ Primer5 刚好看到一半,总结一下C++11里面确实加了很多新东西,如果没有任何了解,别说自己写了,看别人写的代码估计都会有些吃力.C++ Primer5是学习C++1 ...
- C++ 11 新特性
C++11新特性: 1.auto 2.nullptr 3.for 4.lambda表达式 5.override ...
- 在C++98基础上学习C++11新特性
自己一直用的是C++98规范来编程,对于C++11只闻其名却没用过其特性.近期因为工作的需要,需要掌握C++11的一些特性,所以查阅了一些C++11资料.因为自己有C++98的基础,所以从C++98过 ...
- C++11新特性——range for
很多编程语言都有range for语法功能,自C++11起,终于将这个重要功能加入C++标准中.range for语句,可以方便的遍历给定序列中的每个元素并对其执行某种操作. 1.基本语法 for(d ...
- C++11新特性——大括号初始化
C++11之前,C++主要有以下几种初始化方式: //小括号初始化 string str("hello"); //等号初始化 string str="hello" ...
随机推荐
- ubuntu中写一个shell脚本的过程
gedit hello.sh ,然后输入 #!/bin/bash echo "Hello world!" chmod +x hello.sh ./hello.sh
- 《Java虚拟机原理图解》 1.2.3、Class文件里的常量池具体解释(下)
NO9.类中引用到的field字段在常量池中是如何描写叙述的?(CONSTANT_Fieldref_info, CONSTANT_Name_Type_info) 一般而言.我们在定义类的过程中会定义一 ...
- Android应用之——自己定义控件ToggleButton
我们经常会看到非常多优秀的app上面都有一些非常美丽的控件,用户体验非常好.比方togglebutton就是一个非常好的样例,IOS系统以下那个精致的togglebutton现在在android以下也 ...
- nyoj--74--小学生算术(水)
小学生算术 时间限制:3000 ms | 内存限制:65535 KB 难度:1 描述 很多小学生在学习加法时,发现"进位"特别容易出错.你的任务是计算两个三位数在相加时需要多少 ...
- linux + nginx 的配置优化
linux 关于TCP/IP 的优化配置 配置文件/etc/sysctl.conf 修改完文件生效的命令 /sbin/sysctl -p 如下是总结的配置内容及说明 net.ipv4.con ...
- 13. Roman to Integer[E]罗马数字转整数
题目 Given a roman numeral, convert it to an integer. Input is guaranteed to be within the range from ...
- Android 多个APK共享数据
Android给每个APK进程分配一个单独的用户空间,其manifest中的userid就是对应一个Linux用户(Android 系统是基于Linux)的.所以不同APK(用户)间互相访问数据默认是 ...
- sqlserver数据文件位置如何迁移
亲测有效的一种方式: 1.对应的数据库脱机 2.迁移物理文件 3.删除原有实例 4.附加
- IP地址转化为数字,charindex ,SUBSTRING
SET NOCOUNT ON; declare @I_PCity table ( IPStart nvarchar(), Area nvarchar(), CityID int, IPID int ) ...
- http协议以及防盗链技术
http协议,又称为超文本传输协议,顾名思义,http协议不仅能传输文本,还能传输图片,视频,压缩包等文件,http协议是建立在tcp/ip协议的基础之上的,http协议对php程序员来讲可以说是重中 ...