C++ 11中的bind解析

参考自:https://blog.csdn.net/zhouguoqionghai/article/details/45770523

bind函数的返回值是一个函数对象/仿函数, 统一的一致性语法。

1、

bind函数(包装器/适配器)能够将用户提供的需要一个参数的函数对象调整为不需要参数的函数对象。需要的时候,绑定的值(这个例子中是123)存储在函数对象中,自动地传递给用户指定的函数。

f = bind(my_handler, 123)
在这里插入图片描述

一个匿名类(即这里的binder类)的成员变量捕获了函数指针的值和绑定的参数的值,匿名类重载了operator()运算符,从而其对象为仿函数/函数对象。在operator函数体内,使用捕获的函数指针和参数,回调函数指针指向的函数。匿名类只会捕获提前绑定的参数。

当用户调用该函数对象时,实现延迟调用。也就是说,f = bind(my_handler, 123),产生了一个对象,这个f是一个函数对象(仿函数),只有当使用f()的时候,才会去真正的调用其所绑定的可调用对象(这里是函数),这也就是实现了先绑定,需要的时候在延迟调用。

2、

绑定一个参数也可以将类的成员函数调整为不需要参数的函数对象。正如你所知,非静态的成员函数都有一个隐式的this指针参数。这就意味着在函数指针内部需要绑定一个指向该类的一个对象的指针:

f = bind(&session::handler, this)
在这里插入图片描述

匿名类捕获一个指向session的对象的指针作为其成员变量的值。当调用这个函数对象时,即使用f()的时候,实际上就相当于一个session的类对象调用其自己的handler方法!

3、

隐式的this指针可以通过给函数对象传入一个显示的参数:

f = bind(&session::handler, _1)

在这里插入图片描述
绑定的时候有几个占位符,就要传入几个参数。因为没有显示的绑定this,这个时候调用f的时候就需要传入一个session对象。即:
在这里插入图片描述

4、

函数对象通常同时使用提前绑定的参数和调用时提供的参数。这个过程可以通过给成员函数绑定参数来实现:

f = bind(&session::handler, this, _1)
在这里插入图片描述

这里的_1是std::placeholders命名空间里的占位符,在调用函数对象的时候,参数按对应的顺序传递给占位符。

当调用这个函数对象的时候,绑定的时候有几个参数,就要传入几个参数,因为bind上了this,所以传入的就只是session::handler的参数就行了。

5、

当然也可以使用非成员函数,这就简单很多:

f = bind(&session::handler, 123, _1, _2)
在这里插入图片描述

6、

有的时候,函数对象调用时会提供一些不会被目标函数使用到的参数,bind() 函数会自动的忽略这些多余的参数:

f = bind(&session::handler, 123, _1)
在这里插入图片描述

7、

多余的参数不需要在函数签名的最后边:

f = bind(&session::handler, 123, _2)

在这里插入图片描述

也就是说,提前绑定的参数会使调用时又提供的参数无效而丢弃。

8、

bind() 函数允许改变参数的顺序让目标函数能够适应函数对象的函数签名:

f = bind(&session::handler, _2, 123, _1)在这里插入图片描述

从上边的图中可以看出,bind() 函数的原理和lambda函数的原理是类似的,都是通过捕获变量来生成一个匿名的类,在类中重载operator() ,从而其对象是能够调用的函数对象/仿函数。

  • 4
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值