boost::bind的简单实现
前言
在上一篇blog中简单的实现了boost::function,支持带有2个参数的函数/函数指针,函数对象,函数适配器/bind类,以及带有1个参数的成员函数指针。
本文接着来介绍如何实现一个简单的boost::bind。
基本目标如下:
- 支持接收0个参数的函数/函数指针,函数对象。
- 支持接收1个参数的函数/函数指针,函数对象。
- 支持接收2个参数的函数/函数指针,函数对象。
实现
首先,解决占位符的问题:
namespace
{ struct Placeholders1
{
} _1; struct Placeholders2
{
} _2; }
使用匿名namespace的原因是防止不同编译单元中的命名冲突, 让占位符对象只在其所在的编译单元中可见。
在boost::bind源码中主要是通过2个list表维持各种相关信息。一个bindlist表维持了bind接收的绑定参数,包括占位符,用户传入的变量等。一个calllist维持了调用bind返回的对象时所传入的参数信息。它们的通过继承层次的方式来表现的。
下面这个继承层次的每一个类都要作为对应的bindlist和calllist层次中的基类,它们分别保存了bind接收的绑定参数信息(用户传入的变量,占位符),以及调用bind返回的对象时所传入的参数信息。
class Base0
{
}; template<typename T1>
class Base1 : public Base0
{
public:
Base1(T1 data1)
: data1_(data1)
{
} protected:
T1 data1_;
}; template<typename T1, typename T2>
class Base2 : public Base1<T1>
{
public:
Base2(T1 data1, T2 data2)
: Base1<T1>(data1), data2_(data2)
{
} protected:
T2 data2_;
};
接着,就是所谓的calllist的实现了。它们的基类将保存调用bind返回的对象时所传入的参数信息。
class CallList0 : public Base0
{
public:
template<typename _T>
_T operator[](_T arg)
{
return arg;
}
}; template<typename T1>
class CallList1 : public Base1<T1>
{
public:
CallList1(T1 data1)
: Base1<T1>(data1)
{
} T1 operator[](Placeholders1 arg1)
{
return Base1<T1>::data1_;
} template<typename _T>
_T operator[](_T arg)
{
return arg;
} }; template<typename T1, typename T2>
class CallList2: public Base2<T1, T2>
{
public:
CallList2(T1 data1, T2 data2)
: Base2<T1, T2>(data1, data2)
{
} T1 operator[](Placeholders1 arg1)
{
return Base2<T1, T2>::data1_;
} T2 operator[](Placeholders2 arg2)
{
return Base2<T1, T2>::data2_;
} template<typename _T>
_T operator[](_T arg)
{
return arg;
}
};
然后,我们来看看bindlist,它们的基类主要保存了bind接收的占位符、参数信息。
class BindLinst0 : public Base0
{
public:
template<typename Func>
void operator()(Func func)
{
func();
}
}; template<typename T1>
class BindList1 : public Base1<T1>
{
public:
BindList1(T1 data1)
: Base1<T1>(data1)
{
} template<typename Func, typename Call>
void operator()(Func func, Call call)
{
func(call[Base1<T1>::data1_]);
}
}; template<typename T1, typename T2>
class BindList2 : public Base2<T1, T2>
{
public:
BindList2(T1 data1, T2 data2)
: Base2<T1, T2>(data1, data2)
{
} template<typename Func, typename Call>
void operator()(Func func, Call call)
{
func(call[Base2<T1, T2>::data1_], call[Base2<T1, T2>::data2_]);
}
};
接下来就是bind函数所返回的对象了,相信童鞋们可以想象的到,这个对象中应该主要保存的是bind函数接收的参数信息咯,并且他还保存了所注册的函数。
template<typename Func, typename Bind>
class BindImpl
{
public:
BindImpl(Func func, Bind bindlist)
: func_(func), bindlist_(bindlist)
{
} void operator()()
{
bindlist_(func_);
} template<typename T1>
void operator()(T1 data1)
{
bindlist_(func_, CallList1<T1>(data1));
} template<typename T1, typename T2>
void operator()(T1 data1, T2 data2)
{
bindlist_(func_, CallList2<T1, T2>(data1, data2));
} protected:
Func func_;
Bind bindlist_;
};
如此,基本的轮廓就已经出来了。bind函数返回一个BindImpl对象,里面保存了注册的函数和bind接收的占位符、参数信息。当我们调用这个对象的时候,会生成一个calllist对象,它保存了调用BindImpl对象时所传入的参数,然后在bindlist中调用注册的函数。
需要的注意的是,在bindlist调用函数时我们转而调用了calllist的operator[]函数,通过它来判断传入的参数是占位符还是用户传入的参数,如果是占位符,那么就返回calllist中保存的之前注册的用户传入的信息。如果不是占位符,operator[]函数就单纯的返回他接收的参数,也就是之前用户调用BindImpl时传入的参数。
最后,我们通过一组重载的bind函数来实现对接收0个参数、1个参数、2个参数的支持,它们返回的是一个BindImpl对象。
template<typename Func>
BindImpl<Func, BindLinst0> bind(Func func)
{
return BindImpl<Func, BindLinst0>(func, BindLinst0());
} template<typename Func, typename T1>
BindImpl<Func, BindList1<T1> > bind(Func func, T1 data1)
{
return BindImpl<Func, BindList1<T1> >(func, BindList1<T1>(data1));
} template<typename Func, typename T1, typename T2>
BindImpl<Func, BindList2<T1, T2> > bind(Func func, T1 data1, T2 data2)
{
return BindImpl<Func, BindList2<T1, T2> >(func, BindList2<T1, T2>(data1, data2));
}
这样bind函数的基本功能就实现了,但是需要提到的是,目前的bind并不支持注册成员函数。如果要支持成员函数注册的话,需要萃取函数指针的返回类型以及每个参数类型,具体方法在上一篇blog《boost::function的简单实现》中介绍到了,有兴趣的童鞋可以去看看。
下面就来简单的测试一下:
int get(int a, int b)
{
std::cout << a + b << std::endl;
return 0;
} class Point
{
public:
int operator()(int a, int b)
{
std::cout << "Point::operator() called: a = " << a + b << std::endl;
return a+b;
}
}; int main(int argc, char const *argv[])
{
bind(get, _1, 10)(20, 1);
bind(Point(), _1, _2)(3, 4);
return 0;
}
结果为:
30
Point::operator() called: a = 7
得到的结果正如预期的一样。
参考文献
- boost中文手册. bind.hpp
(完)
boost::bind的简单实现的更多相关文章
- 使用BOOST BIND库提高C++程序性能
Boost.Bind为函数和函数对象,值语义和指针提供语义了一致的语法.我们首先通过一些简单的例子来看看它的基本用法,之后我们会延伸到嵌套绑定以实现功能组合.理解bind用法的一个关键是理解占位符(p ...
- boost::bind 和 boost::function 基本用法
这是一篇介绍bind和function用法的文章,起因是近来读陈硕的文章,提到用bind和function替代继承,于是就熟悉了下bind和function的用法,都是一些网上都有的知识,记录一下,期 ...
- (转)boost::bind介绍
转自:http://www.cnblogs.com/sld666666/archive/2010/12/14/1905980.html 这篇文章介绍boost::bind()的用法, 文章的主要内容是 ...
- [置顶] 编程模仿boost::function和boost::bind
boost::function和boost::bind结合使用是非常强大的,他可以将成员函数和非成员函数绑定对一个对象上,实现了类似C#的委托机制.委托在许多时候可以替代C++里面的继承,实现对象解耦 ...
- 手把手教你实现boost::bind
前言 boost::bind操作想必大家都使用过,它特别神奇,能够绑定函数与参数,绑定后能够改变参数数量,并且还可以使用占位符.它可以绑定普通函数也可以绑定类成员函数.好多小伙伴试图看过boost:: ...
- boost::bind 介绍
boost::bind 介绍 这篇文章介绍boost::bind()的用法, 文章的主要内容是参考boost的文档. 1. 目的 boost::bind 是std::bindlist 和 std: ...
- Boost::bind使用详解
1.Boost::bind 在STL中,我们经常需要使用bind1st,bind2st函数绑定器和fun_ptr,mem_fun等函数适配器,这些函数绑定器和函数适配器使用起来比较麻烦,需要根据是全局 ...
- boost::bind 学习
最近学习了太多与MacOS与Iphone相关的东西,因为不会有太多人有兴趣,学习的平台又是MacOS,不太喜欢MacOS下的输入法,所以写下来的东西少了很多. 等我学习的东西慢慢的与平台无关的时 ...
- boost::function的简单实现
前言 boost::function和boost:bind是一对强大的利器.相信用过的童鞋多少有些体会. 虽然平时在用boost::function,但是用的时候心中总会一些不安,因为不知道它是怎么实 ...
随机推荐
- LightGBM的算法介绍
LightGBM算法的特别之处 自从微软推出了LightGBM,其在工业界表现的越来越好,很多比赛的Top选手也掏出LightGBM上分.所以,本文介绍下LightGBM的特别之处. LightGBM ...
- ISAP 最大流 最小割 模板
虽然这道题用最小割没有做出来,但是这个板子还是很棒: #include<stdio.h> #include<math.h> #include<string.h> # ...
- xampp开户,apache打开出现端口被占用提示
刚装上去的时候,可以打开xampp,但是重启的时候出现以后以下问题 13:49:02 [Apache] Error: Apache shutdown unexpectedly.13:49:0 ...
- 【python】实现一个python编程的小时钟!
[本实验内容] 1.GUI.PyQT5介绍2.实现此次实验效果 [一 GUI.PyQt5介绍] 1.Python简介 2.GUI介绍 几个常用的Python GUI库: (1)wxPython (2) ...
- vue中表单的动态绑定
有一个表单: <el-form :model="configForm"> </el-form> 如果configForm初始化为{},此对象是动态变化的,如 ...
- flask-sqlalchemy 用法总结
Flask-SQLAlchemy是一个Flask扩展,能够支持多种数据库后台,我们可以不需要关心SQL的处理细节,操作数据库,一个基本关系对应一个类,而一个实体对应类的实例对象.Flask是一个轻量级 ...
- springboot 实现自定义注解
1.定义一个注解@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface T ...
- BZOJ5340 [Ctsc2018]假面 【概率dp】
题目链接 BZOJ5340 题解 我们能很容易维护每个人当前各种血量的概率 设\(p[u][i]\)表示\(u\)号人血量为\(i\)的概率 每次攻击的时候,讨论一下击中不击中即可转移 是\(O(Qm ...
- C++——设计与演化——读书笔记
<<c++设计与演化>>1.c++的保护模式来自于访问权限许可和转让的概念; 初始化和赋值的区分来自于转让能力的思考; c++的const概念是从读写保护机制中演化出来. 2. ...
- bootstrap和bootstrap-select去除蓝色边框outline
/*bootstrap outline设置*/ .btn:focus, .btn:active:focus, .btn.active:focus, .btn.focus, .btn:active.fo ...