前言

在上一篇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. LightGBM的算法介绍

    LightGBM算法的特别之处 自从微软推出了LightGBM,其在工业界表现的越来越好,很多比赛的Top选手也掏出LightGBM上分.所以,本文介绍下LightGBM的特别之处. LightGBM ...

  2. ISAP 最大流 最小割 模板

    虽然这道题用最小割没有做出来,但是这个板子还是很棒: #include<stdio.h> #include<math.h> #include<string.h> # ...

  3. xampp开户,apache打开出现端口被占用提示

    刚装上去的时候,可以打开xampp,但是重启的时候出现以后以下问题 13:49:02  [Apache]     Error: Apache shutdown unexpectedly.13:49:0 ...

  4. 【python】实现一个python编程的小时钟!

    [本实验内容] 1.GUI.PyQT5介绍2.实现此次实验效果 [一 GUI.PyQt5介绍] 1.Python简介 2.GUI介绍 几个常用的Python GUI库: (1)wxPython (2) ...

  5. vue中表单的动态绑定

    有一个表单: <el-form :model="configForm"> </el-form> 如果configForm初始化为{},此对象是动态变化的,如 ...

  6. flask-sqlalchemy 用法总结

    Flask-SQLAlchemy是一个Flask扩展,能够支持多种数据库后台,我们可以不需要关心SQL的处理细节,操作数据库,一个基本关系对应一个类,而一个实体对应类的实例对象.Flask是一个轻量级 ...

  7. springboot 实现自定义注解

    1.定义一个注解@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface T ...

  8. BZOJ5340 [Ctsc2018]假面 【概率dp】

    题目链接 BZOJ5340 题解 我们能很容易维护每个人当前各种血量的概率 设\(p[u][i]\)表示\(u\)号人血量为\(i\)的概率 每次攻击的时候,讨论一下击中不击中即可转移 是\(O(Qm ...

  9. C++——设计与演化——读书笔记

    <<c++设计与演化>>1.c++的保护模式来自于访问权限许可和转让的概念; 初始化和赋值的区分来自于转让能力的思考; c++的const概念是从读写保护机制中演化出来. 2. ...

  10. bootstrap和bootstrap-select去除蓝色边框outline

    /*bootstrap outline设置*/ .btn:focus, .btn:active:focus, .btn.active:focus, .btn.focus, .btn:active.fo ...