C++ 11中的bind解析

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

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

1、

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

f = bind(my_handler, 123)
C++ 11中的bind解析

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

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

2、

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

f = bind(&session::handler, this)
C++ 11中的bind解析

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

3、

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

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

C++ 11中的bind解析
绑定的时候有几个占位符,就要传入几个参数。因为没有显示的绑定this,这个时候调用f的时候就需要传入一个session对象。即:
C++ 11中的bind解析

4、

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

f = bind(&session::handler, this, _1)
C++ 11中的bind解析

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

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

5、

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

f = bind(&session::handler, 123, _1, _2)
C++ 11中的bind解析

6、

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

f = bind(&session::handler, 123, _1)
C++ 11中的bind解析

7、

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

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

C++ 11中的bind解析

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

8、

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

f = bind(&session::handler, _2, 123, _1)C++ 11中的bind解析

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

上一篇:ubuntu apt-get 遇到的问题


下一篇:Springboot中使用线程拿到当前账号信息