前言

在上一篇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

得到的结果正如预期的一样。

参考文献

  1. boost中文手册. bind.hpp

(完)

boost::bind的简单实现的更多相关文章

  1. 使用BOOST BIND库提高C++程序性能

    Boost.Bind为函数和函数对象,值语义和指针提供语义了一致的语法.我们首先通过一些简单的例子来看看它的基本用法,之后我们会延伸到嵌套绑定以实现功能组合.理解bind用法的一个关键是理解占位符(p ...

  2. boost::bind 和 boost::function 基本用法

    这是一篇介绍bind和function用法的文章,起因是近来读陈硕的文章,提到用bind和function替代继承,于是就熟悉了下bind和function的用法,都是一些网上都有的知识,记录一下,期 ...

  3. (转)boost::bind介绍

    转自:http://www.cnblogs.com/sld666666/archive/2010/12/14/1905980.html 这篇文章介绍boost::bind()的用法, 文章的主要内容是 ...

  4. [置顶] 编程模仿boost::function和boost::bind

    boost::function和boost::bind结合使用是非常强大的,他可以将成员函数和非成员函数绑定对一个对象上,实现了类似C#的委托机制.委托在许多时候可以替代C++里面的继承,实现对象解耦 ...

  5. 手把手教你实现boost::bind

    前言 boost::bind操作想必大家都使用过,它特别神奇,能够绑定函数与参数,绑定后能够改变参数数量,并且还可以使用占位符.它可以绑定普通函数也可以绑定类成员函数.好多小伙伴试图看过boost:: ...

  6. boost::bind 介绍

    boost::bind 介绍   这篇文章介绍boost::bind()的用法, 文章的主要内容是参考boost的文档. 1. 目的 boost::bind 是std::bindlist 和 std: ...

  7. Boost::bind使用详解

    1.Boost::bind 在STL中,我们经常需要使用bind1st,bind2st函数绑定器和fun_ptr,mem_fun等函数适配器,这些函数绑定器和函数适配器使用起来比较麻烦,需要根据是全局 ...

  8. boost::bind 学习

    最近学习了太多与MacOS与Iphone相关的东西,因为不会有太多人有兴趣,学习的平台又是MacOS,不太喜欢MacOS下的输入法,所以写下来的东西少了很多.    等我学习的东西慢慢的与平台无关的时 ...

  9. boost::function的简单实现

    前言 boost::function和boost:bind是一对强大的利器.相信用过的童鞋多少有些体会. 虽然平时在用boost::function,但是用的时候心中总会一些不安,因为不知道它是怎么实 ...

随机推荐

  1. Week1 Team Homework #1 from Z.XML-总结学长经验教训

    谭传奇学长: 我们的弯路可能是,一开始没有从最基础的部分开始迭代开发,一开始就想的太远了一些,每一步开的有点太大了,所以可能有些东西最后就连不上,也没有能够按时完成.如果可以先做出一个能用的版本,然后 ...

  2. 远程sql数据库连接不上,provider: 命名管道提供程序, error: 40 - 无法打开到 SQL Server 的连接 错误解决

    错误信息: “ 标题: 连接到服务器------------------------------ 无法连接到 192.168.1.20. ------------------------------其 ...

  3. java正则表达式2 -- 匹配、切割、查找

    import java.util.Arrays; /* 正则表达式的作用: 1 匹配 2 切割 3 替换 * */ public class Demo1 { public static void ma ...

  4. Java循环控制语句-switch

    Java循环控制语句之一switch 不同于其他循环控制语句的特性: switch的英文解释为开关,正如它的解释一样,switch循环的特点就像开关一样,跳到哪一个条件即会出现某一种结果. 写法: s ...

  5. [剑指Offer] 21.栈的压入、弹出序列

    题目描述 输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否为该栈的弹出顺序.假设压入栈的所有数字均不相等.例如序列1,2,3,4,5是某栈的压入顺序,序列4,5,3,2,1是该压栈序 ...

  6. SQL Server Profiler的简单使用,方便查找和发现SQL执行的效率和语句问题

    1 打开Server Profiler 2 去掉不必要的干扰,数据库的连接和断开之类的 3. 选择“显示所有列”,之后在列表中,勾选“DatabaseName”项. 4设置筛选器,这里设置只是过滤数据 ...

  7. servletContext的定义

  8. hihocoder 1323 回文字符串(字符串+dp)

    题解: 比较水的题目 dp[i][j]表示[i...j]最少改变几次变成回文字符串 那么有三种转移 dp[i][j] = dp[i+1][j-1] + s[i] != s[j] dp[i][j] = ...

  9. SRM708 div1 PalindromicSubseq(动态规划+容斥原理)

    题目大意:给定一个字符串,记X[i]为包含s[i]这个字符的所有子列是回文串的个数(注意是子列而不是子串),求出所有的X[i]*(i+1),然后异或起来作为返回结果 题解: 首先用容斥来想,如果当前枚 ...

  10. [CF1031E]Triple Flips

    题目大意:给你一个长度为$n$的$01$串,一次操作定义为:选取$3$个等距的元素,使其$0$变$1$,$1$变$0$,要求在$\Big\lfloor \dfrac n 3\Big\rfloor+12 ...