1. std::function

(1)首先是一个类模板用于包装可调用对象。可以容纳除了类成员(函数)指针之外的所有可调用对象。

(2)可以将普通函数,lambda表达式和函数对象类统一起来。尽管它们并不是相同的类型,但通过function类模板,可以转化为相同类型的对象(function对象),这样就可以用统一的方式来保存或传递可调用对象

(3)实现了一套类型消除机制,可用统一的方式处理不同类型的可调用对象

(4)std::function进一步深化以数据为中心(封装)的面向对象思想(连函数都对象化了)

【编程实验】std::function作为函数的入参“万能类型”

#include <iostream>
#include <functional> //for std::bind & std::function using namespace std;
using namespace std::placeholders; //传统C函数
int func(int a, int b)
{
return a + b;
} //仿函数
class Functor
{
public:
int operator()(int a, int b)
{
return a - b;
}
}; //类的成员函数
class Foo
{
public:
static int func(int a, int b)
{
return a * b;
} int func_common(int a, int b)
{
return a * b;
}
}; //模板函数
template < typename T>
auto template_func (T a ,T b)->decltype(a + b)
{
return a + b;
} //测试函数
//1. 这是一个“万能”的函数,可接受不同类型的可调用对象,而不必为他们重载多个版本的test函数
//2. 可以测试的对象包含各类可调用对象,如普通函数、仿函数、lambda表达式等
int test(int x, int y, const std::function<int(int, int)>& callableObjects)
{
return callableObjects(x, y);
} int main()
{
//传入普通函数
using Fn = int(*)(int, int);
Fn fn = &func;
//decltype(func)* fn = &func;
cout << "func(3, 4): " << test(, , fn) << endl; //普通函数指针可以直接赋值给std::function //测试函数模板
auto tfn = template_func<int>;
cout << "template_func<int>(3, 4): " << test(, , tfn) << endl; //测试仿函数
Functor ftor;
cout << "Functor(3, 4): " << test(, , ftor) << endl; //仿函数可以直接赋值给std::function //测试lambda表达式
auto lbd = [](int a, int b){ return a + b; };
//lambda表达式可以直接赋值给std::function
cout << "[](int a, int b){ return a + b }: " << test(, , lbd) << endl; //测试类的成员函数
auto memfn = Foo::func; //静态成员函数
//静态成员函数可以直接赋值给std::function
cout << "Foo::func(3, 4): " << test(, , memfn) << endl;
//普通成员函数(不能直接赋值给std::function,需先经bind转成std::function)
auto commfn = bind(&Foo::func_common, Foo(), _1, _2);//先转换为std::function
cout << "Foo::func_common(3, 4): " << test(, , commfn) << endl; return ;
}
/*测试结果
e:\Study\C++11\12>g++ -std=c++11 test1.cpp
e:\Study\C++11\12>a.exe
func(3, 4): 7
template_func<int>(3, 4): 7
Functor(3, 4): -1
[](int a, int b){ return a + b }: 7
Foo::func(3, 4): 12
Foo::func_common(3, 4): 12
*/

2. std::function和std::bind的关系

(1)std::bind是一个函数模板用于将可调用对象及其参数一起,绑定成一个std::function对象。其返回值是个std::function类型。

(2)std::function是一个类模板,用来包装各类可调用对象为新的callable object。他可以接受全局函数、类的静态成员函数并直接进行封装。但不能直接接受类的非静态成员,需要使用bind绑定才能赋值给std::function。

【编程实验】利用function+bind实现回调函数类似于函数指针的作用,可保存、延迟处理函数

#include <iostream>
#include <functional> //for std::bind & std::function using namespace std;
using namespace std::placeholders; //操作系统
class OperatingSystem
{
private:
using NotifyFunc = std::function<void(string, string)>;
NotifyFunc m_callback;
public:
string jobname;
string jobmessage;
public:
OperatingSystem() : m_callback(nullptr){}; template <typename T1, typename T2>
void RegisterNotify(T1 memberFunc, T2* pThis)
{
m_callback = std::bind(memberFunc, pThis, _1, _2);
} template <typename T>
void RegisterNotify(T globalFunc)
{
m_callback = std::bind(globalFunc, _1, _2);
} //当作业加入时,会回调作业本身定义的个性化通知!
bool Notify()
{
if(m_callback != NULL)
m_callback(jobname, jobmessage); return false;
}
}; //作业类
class Job
{
private: //每次作业加入进来,会有个性化的通知
void SendMsg(string name, string msg)
{
cout <<"(local)"<< name << ": " << msg << endl;
} public:
string name;
string msg;
Job(string name, string msg):name(name),msg(msg)
{
} void addJob(OperatingSystem& os)
{
os.RegisterNotify(&Job::SendMsg, this);
os.jobname = name;
os.jobmessage = msg;
os.Notify();
}
}; void SendMsg(string name, string msg)
{
cout << "(global)" <<name << ": " << msg << endl;
} int main()
{
OperatingSystem os; //绑定Job类的成员函数
Job job1("job1", "hello world!");
job1.addJob(os); Job job2("job2", "thank you!");
job2.addJob(os); //绑定全局函数
os.RegisterNotify(&SendMsg);
os.jobname = "job3";
os.jobmessage = "nice to meet you!";
os.Notify(); return ;
}
/*输出结果
e:\Study\C++11\12>g++ -std=c++11 test2.cpp
e:\Study\C++11\12>a.exe
(local)job1: hello world!
(local)job2: thank you!
(global)job3: nice to meet you!
*/

第12课 std::bind和std::function(3)_std::function可调用对象包装器的更多相关文章

  1. std::bind和std::function

    std::bind 用于绑定一个函数,返回另外一种调用方式的函数对象 ,可以改变参数顺序 和个数,特别是在多线程的程序中,经常用它将函数进行包装,然后打包发送给工作线程,让工作线程去执行我们的任务. ...

  2. c++11特性与cocos2d-x 3.0之std::bind与std::function

    昨天同事让帮忙写一小功能,才发现cocos2d-x 3.0 和 cocos2d-x 3.0rc0 差别还是相当大的. 发现Label这一个控件,3.0就比rc0版本多了一个创建函数,更为关键的是3.0 ...

  3. 第11课 std::bind和std::function(2)_std::bind绑定器

    1. 温故知新:std::bind1st和std::bind2nd (1)bind1st.bind2nd首先它们都是函数模板,用于将参数绑定到可调用对象(如函数.仿函数等)的第1个或第2个参数上. ( ...

  4. 第10课 std::bind和std::function(1)_可调用对象

    1. 几种可调用对象(Callable Objects) (1)普通函数指针或类成员的函数指针 (2)具有operator()成员函数的类对象(仿函数).如c++11中的std::function类模 ...

  5. std::bind与std::ref, why and how

    首先解释下为什么有时候需要bind. 我们可以用bind从函数T add(T a, T b)造出个inc()来,即把b写死为1.这个例子本身比较傻,但有不傻的应用. template<typen ...

  6. std::function"函数"对象包装器

    语义: 类模板std::function是可调用对象的包装器,可以包装除了类成员之外的所有可调用对象.包括,普通函数,函数指针,lambda,仿函数.通过指定的模板参数,它可以用统一的方式保存,并延迟 ...

  7. 【浅析C++11】std::function和std::bind

    目录 std::function可调用对象包装器 std::function基本用法 std::function/std::bind与抽象工厂.工厂方法的一点思考 std::function可调用对象 ...

  8. std::bind接口与实现

    前言 最近想起半年前鸽下来的Haskell,重温了一下忘得精光的语法,读了几个示例程序,挺带感的,于是函数式编程的草就种得更深了.又去Google了一下C++与FP,找到了一份近乎完美的讲义,然后被带 ...

  9. C++11 std::bind std::function 高级使用方法

    从最基础的了解,std::bind和std::function /* * File: main.cpp * Author: Vicky.H * Email: eclipser@163.com */ # ...

随机推荐

  1. pnpm 快速节省磁盘工具的包管理工具

    nodejs 相关的包管理工具有很多,我们常用的有 npm cnpm(我基本已经不用了),yarn... pnpm 是另外一个不错的包管理工具,包含以下特性 快速 节省空间,一个版本的包只会在磁盘中存 ...

  2. 普林斯顿数学指南(第一卷) (Timothy Gowers 著)

    第I部分 引论 I.1 数学是做什么的 I.2 数学的语言和语法 I.3 一些基本的数学定义 I.4 数学研究的一般目的 第II部分 现代数学的起源 II.1 从数到数系 II.2 几何学 II.3 ...

  3. Vue学习入门

    1.安装WebStorm: 2.激活WebStorm:https://blog.csdn.net/qq_40147863/article/details/81317709 3.安装全局脚手架:npm ...

  4. What is the difference between concurrency, parallelism and asynchronous methods?

    Ref: http://stackoverflow.com/questions/4844637/what-is-the-difference-between-concurrency-paralleli ...

  5. 利用pandas将numpy数组导出生成excel

    代码 # -*- coding: utf- -*- """ Created on Sun Jun :: @author: Bruce Lau ""&q ...

  6. Hadoop HDFS NameNode工作机制

    Secondary namenode 首先,我们假设如果存储在Namenode节点的磁盘中,因为经常需要进行随机访问,还有响应客户请求,必然是效率过低.因此,元数据需要存放在内存中.但如果只存在内存中 ...

  7. 让邮件服务器发出的Email不被认为是垃圾邮件

    配置一个Email服务器很简单,用Postfix等软件稍微配置一下就可以了,几分钟搞定. 但Email服务器发出去的Email很可能经常被人当成垃圾邮件,怎么配置Email服务器才能让Email服务器 ...

  8. 自动加载以及Composer的实现

    类的自动加载 两个函数 __autoload()魔术方法.spl_autoload_register 当php文件中使用了new关键字实例化一个对象时,如果该类没有在本php文件中被定义,将会触发__ ...

  9. windows环境下把Python代码打包成独立执行的exe

    windows环境下把Python代码打包成独立执行的exe可执行文件   有时候因为出差,突然急需处理一批数据.虽然写好的脚本存储在云端随用随取,然而编译的环境还需要重新搭建,模块也需要重新装载,从 ...

  10. Mac OS安装Windows各版本时注意事项(2014年后的Mac机相信会有这些问题)

    2014年后的Mac Mini安装Windows时候,会遇上更种坑,我这几天不断失败及尝试更种版本,各有心得体会,我写下来是为了避免大家遇到像我这种问题. (2014年之前的Mac Mini是否这样, ...