文档章节

C++ 实现简单的闭包

兔之
 兔之
发布于 2016/12/08 11:15
字数 591
阅读 706
收藏 14

这里的闭包指的是一个闭包类,这个闭包类使用一个通用的辅助接口函数(如 NewCallback)来生成。在生成的时候传入需要使用的函数指针以及这个函数将要使用到的参数,生成闭包工具类后再使用一个通过的辅组函数(如 Run)让函数指针执行。闭包将函数和参数封装到类中,在一定的场景下进行使用,这就是闭包的作用。

示例

#include <iostream>
#include <string>

class Closure {
public:
    Closure() {}
    ~Closure() {}

    virtual void Run() = 0; //纯虚函数
};

template <typename Arg1>
class FunctionClosure1 : public Closure {
public:
    typedef void (*FunctionType)(Arg1);

    FunctionClosure1(FunctionType f, Arg1 arg1) :
        _f(f),
        _arg1(arg1) {
        }
    ~FunctionClosure1() {
    }

    virtual void Run() {
        _f(_arg1);
        delete this;
    }
private:
    FunctionType _f;
    Arg1 _arg1;
};

template <typename Arg1>
Closure* NewCallback(void(*function)(Arg1), Arg1 arg1) {
    return new FunctionClosure1<Arg1>(function, arg1);
}

// 带一个参数的函数
template<typename type>
void foo(type data) 
{
    std::cout << "foo data=" << data << std::endl;
}

int main()
{
    Closure* closure;
    closure = NewCallback(foo<std::string>, std::string("titus"));
    //等价于 closure = new FunctionClosure1<std::string>(foo<std::string>, std::string("titus"));
    closure->Run();
    //自己释放 delete closure
    return 0;
}

Closure 定义为纯虚类,不能实例化,必须由子类实现它的虚函数后再才能实例化。

FunctionClosure1 为 Closure 的子类,定义为模版类,可以定制传入参数的类型。它有两个私有成员,函数指针成员 _f,参数 _arg1,在成员方法 Run 中会让函数 _f 传入参数 _arg1 进行调用。而函数指针成员是在类初始化时传入的,相当于函数也是可以定制的。运行完之后 delete this,这是因为 NewCallback 在堆上 new 了一个对象,这里自动进行资源释放,当然也可以自己释放。

NewCallback 是一个辅助函数,用来生成子闭包类,它需要传入函数指针和参数。

在使用时,父指针指向 NewCallback创建的子类,同时传入函数指针和参数,最后调用子类继承实现的 Run 方法。

这么看来,闭包可以看成是对回调函数的封装。

标准库写法

头文件引入 functional 标准库,用 C++11 的写法也可以实现上述例子

std::function<void(string)> std_closure=foo;
std_closure(string("test std"));

使用 lambda 表达式

std::function<void()> std_closure = []() {foo(string("test lambda"));};
std_closure();

参考

http://www.cnblogs.com/Aion/p/3449756.html

http://yingshin.github.io/c/cpp/2015/12/01/closure-implement-in-C++

http://www.cnblogs.com/lvpengms/archive/2011/02/21/1960078.html

© 著作权归作者所有

共有 人打赏支持
兔之
粉丝 66
博文 247
码字总数 95896
作品 7
深圳
程序员
加载中

评论(6)

兔之
兔之

引用来自“Windoze”的评论

还不如这样:
`std::function<void()> closure=[=](){ foo(some_string_from_outside); };`
可以这么写,这篇文章相当于简单实现下类 function 的功能。
Windoze
Windoze
还不如这样:
`std::function<void()> closure=[=](){ foo(some_string_from_outside); };`
兔之
兔之

引用来自“magiclogy”的评论

无论是函数对象还是lambda都比这个靠谱吧?
函数对象的函数计算逻辑是写在 () 运算符里,闭包类的函数是从外面传入,后者具有更好的封装性。
兔之
兔之

引用来自“magiclogy”的评论

无论是函数对象还是lambda都比这个靠谱吧?
和 lambda 表达式一样,异步回调。
阿小莫
阿小莫
看了3遍我还没搞明白这种实现方式应用在什么场景?
m
magiclogy
无论是函数对象还是lambda都比这个靠谱吧?
深入分析golang多值返回以及闭包的实现

一、前言 golang有很多新颖的特性,不知道大家的使用的时候,有没想过,这些特性是如何实现的?当然你可能会说,不了解这些特性好像也不影响自己使用golang,你说的也有道理,但是,多了解底...

万建宁
07/17
0
0
C++11 中值得关注的几大变化

源文章来自前C++标准委员会的 Danny Kalev 的 The Biggest Changes in C++11 (and Why You Should Care),赖勇浩做了一个中文翻译在这里。所以,我就不翻译了,我在这里仅对文中提到的这些变...

雅各宾
2014/01/17
0
0
【转载】数据结构利器之私房STL

数据结构利器之私房STL 此系列的文章适合初学有意剖析STL和欲复习STL的同学们。 学过c++的同学相信都有或多或少接触过STL。STL不仅仅是c++中很好的编程工具(这个词可能有点歧义,用类库更恰...

悠米海
2012/12/02
0
0
iOS学习笔记(十六)——数据库操作(使用FMDB)

iOS中原生的SQLite API在使用上相当不友好,在使用时,非常不便。于是,就出现了一系列将SQLite API进行封装的库,例如FMDB、PlausibleDatabase、sqlitepersistentobjects等,FMDB (https:/...

sea_god
2014/08/19
0
0
[开源世界]从自动导出动态链接库接口看C++的缺点

自动导出动态链接库接口在C++编程中绝对是一件烦人的事情,因为你不得不大量的重复以下几个步骤: 1.加载动态链接库 2.定义导出函数指针定义 3.定义导出函数指针变量 4.从动态链接库中导出函...

梁欢
2013/10/21
0
2

没有更多内容

加载失败,请刷新页面

加载更多

下一页

深夜胡思乱想

魔兽世界 最近魔兽世界出了新版本, 周末两天升到了满级,比之前的版本体验好很多,做任务不用抢怪了,不用组队打怪也是共享拾取的。技能简化了很多,哪个亮按哪个。 运维 服务器 产品 之间的...

Firxiao
7分钟前
0
0
MySQL 8 在 Windows 下安装及使用

MySQL 8 带来了全新的体验,比如支持 NoSQL、JSON 等,拥有比 MySQL 5.7 两倍以上的性能提升。本文讲解如何在 Windows 下安装 MySQL 8,以及基本的 MySQL 用法。 下载 下载地址 https://dev....

waylau
41分钟前
0
0
微信第三方平台 access_token is invalid or not latest

微信第三方开发平台code换session_key说的特别容易,但是我一使用就带来无穷无尽的烦恼,搞了一整天也无济于事. 现在记录一下解决问题的过程,方便后来人参考. 我遇到的这个问题搜索了整个网络也...

自由的开源
今天
0
0
openJDK之sun.misc.Unsafe类CAS底层实现

注:这篇文章参考了https://www.cnblogs.com/snowater/p/8303698.html 1.sun.misc.Unsafe中CAS方法 在sun.misc.Unsafe中CAS方法如下: compareAndSwapObject(java.lang.Object arg0, long a......

汉斯-冯-拉特
今天
2
0
设计模式之五 责任链模式(Chain of Responsibility)

一. 场景 相信我们都有过这样的经历; 我们去职能部门办理一个事情,先去了A部门,到了地方被告知这件事情由B部门处理; 当我们到了B部门的时候,又被告知这件事情已经移交给了C部门处理; ...

JackieRiver
今天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部