说是延时求值,注意还是想搞一搞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)的更多相关文章

  1. C++11新特性(1) 右值引用

    在C++中,左值(lvalue)是能够获取其地址的一个量.因为常常出如今赋值语句的左边.因此称之为左值.比如一个有名称的变量. 比如: int a=10; //a就是一个左值. 传统的C++引用,都是 ...

  2. [转载] C++11新特性

    C++11标准发布已有一段时间了, 维基百科上有对C++11新标准的变化和C++11新特性介绍的文章. 我是一名C++程序员,非常想了解一下C++11. 英文版的维基百科看起来非常费劲,而中文版维基百 ...

  3. C++ 11学习和掌握 ——《深入理解C++ 11:C++11新特性解析和应用》读书笔记(一)

    因为偶然的机会,在图书馆看到<深入理解C++ 11:C++11新特性解析和应用>这本书,大致扫下,受益匪浅,就果断借出来,对于其中的部分内容进行详读并亲自编程测试相关代码,也就有了整理写出 ...

  4. C++11新特性总结 (二)

    1. 范围for语句 C++11 引入了一种更为简单的for语句,这种for语句可以很方便的遍历容器或其他序列的所有元素 vector<int> vec = {1,2,3,4,5,6}; ...

  5. C++11新特性总结 (一)

    1. 概述 最近在看C++ Primer5 刚好看到一半,总结一下C++11里面确实加了很多新东西,如果没有任何了解,别说自己写了,看别人写的代码估计都会有些吃力.C++ Primer5是学习C++1 ...

  6. C++ 11 新特性

    C++11新特性:          1.auto          2.nullptr          3.for          4.lambda表达式          5.override ...

  7. 在C++98基础上学习C++11新特性

    自己一直用的是C++98规范来编程,对于C++11只闻其名却没用过其特性.近期因为工作的需要,需要掌握C++11的一些特性,所以查阅了一些C++11资料.因为自己有C++98的基础,所以从C++98过 ...

  8. C++11新特性——range for

    很多编程语言都有range for语法功能,自C++11起,终于将这个重要功能加入C++标准中.range for语句,可以方便的遍历给定序列中的每个元素并对其执行某种操作. 1.基本语法 for(d ...

  9. C++11新特性——大括号初始化

    C++11之前,C++主要有以下几种初始化方式: //小括号初始化 string str("hello"); //等号初始化 string str="hello" ...

随机推荐

  1. jquery源码ajax分析

    http://www.cnblogs.com/aaronjs/p/3683925.html

  2. h5-登录

    <html> <head> <meta http-equiv="Content-Type" content="text/html; char ...

  3. hdoj--2579--Dating with girls(2)(搜索+三维标记)

    Dating with girls(2) Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Oth ...

  4. checkbox改写

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  5. linux压缩(解压缩)命令详解

    一.tar命令          tar可以为文件和目录创建档案.利用tar,用户可以为某一特定文件创建档案(备份文件),也可以在档案中改变文件,或者向档案中加入新的文件.tar 最初被用来在磁带上创 ...

  6. 开发辅助 | 前端开发工程师必懂的 UI 知识

           移动 UI 设计的世界 ... 1.屏幕尺寸 屏幕大小,指屏幕对角线的长度,而不是屏幕的宽度或高度: 单位为英寸 如 iPhone 7 屏幕尺寸为 4.7 英寸:三星 S6 屏幕尺寸为 ...

  7. python 3.x 学习笔记17(协程以及I/O模式)

    1.协程(微线程)协程是一种用户态的轻量级线程.协程拥有自己的寄存器上下文和栈.协程调度切换时,将寄存器上下文和栈保存到其他地方,在切回来的时候,恢复先前保存的寄存器上下文和栈.因此: 协程能保留上一 ...

  8. R 连接DB2数据库

    1.odbc文件下载 教程: http://dasapp.oregon.gov/datamart/files/IBM_DB2_9.7_Run_Time_client_Notes.pdf 驱动地址: h ...

  9. NSRunloop总结

    NSRunloop是一个消息处理机制:是一个循环. 系统通过消息队列和runloop与进程(线程)通信. runloop是一个机制和体系结构. 它包含以下几个方面: 1.事件源管理: 2.事件的检索与 ...

  10. ZBrush软件特性之Stencil模板调控板

    在ZBrush中使用Stencil模板我们了解的所有绘图工具,通过它确定模板周围与涂画或模型的位置.本文将详解ZBrush®中如何使用“曲线板”自定义形状. 使用模版 ZBrush模版的作用象我们了解 ...