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,但是用的时候心中总会一些不安,因为不知道它是怎么实 ...
随机推荐
- 九度OJ--Q1166
import java.text.DecimalFormat;import java.util.Scanner; /* * 题目描述: * 立方根的逼近迭代方程是 y(n+1) = y(n)*2/3 ...
- Visual Studio 2012安装包
点击下载
- IDEA使用maven构建时控制台中文乱码的解决办法
使用maven clean install 项目时控制台中文乱码,解决办法如下: Setting->maven->runner VMoptions: -Dfile.encoding=UTF ...
- 3GPP规范命名规则解读
http://blog.sina.com.cn/s/blog_6b10255301012co6.html 学习了解电信技术知识的一个很好的手段是阅读3GPP的规范.但是3GPP有大量的规范,我们可能经 ...
- B - 寻找M
B - 寻找M Time Limit: 1000/1000MS (C++/Others) Memory Limit: 65536/65536KB (C++/Others) Problem Descri ...
- 【历史】- Unix英雄传:图文细数十五位计算机先驱
Unix,一款多任务多用户操作系统,最早由AT&T公司员工及合作伙伴在贝尔实验室于1969年开发完成.Unix的衍生及克隆版本包括Berkeley Unix.Minix.Linux.AIX.A ...
- [剑指Offer] 12.数值的整数次方
[思路1]递归 class Solution { public: double Power(double base, int exponent) { ){ /base; exponent = -exp ...
- 获取src下的文件
- 机器学习基础知识笔记(一)-- 极大似然估计、高斯混合模型与EM算法
似然函数 常说的概率是指给定参数后,预测即将发生的事件的可能性.拿硬币这个例子来说,我们已知一枚均匀硬币的正反面概率分别是0.5,要预测抛两次硬币,硬币都朝上的概率: H代表Head,表示头朝上 p( ...
- [洛谷P1536]村村通
题意:多组数据,当n为0时结束,每组数据表示有n个村子,m条路,求还需要建多少条路,使得所有的村子联通题解:用并查集求出有多少个联通块,然后求解 C++ Code: #include<cstdi ...