Boost库bind函数的嵌套调用
问题提出:
我写了一个函数,它需要使用一个生成整数随机数的随机数发生器作为参数。然后写了一个函数来根据参数生成不同分布的随机数发生器,供前一个函数使用。我调用了boost库的一些基于特定概率分布的随机数生成函数,但是有很多函数的结果是double型的。
我的函数将随机数发生器定义为boost::function<int()>类型。很明显那些生成double的函数不能直接通过bind operator()的方式得到,而需要类型转换。而这个使用bind/function的嵌套调用就比较麻烦了。
假设我有两个函数:
[cpp]
int trans(double);
double gen();
int trans(double);
double gen();
我希望的结果是得到一个函数对象:它使用gen生成原始数字,然后用trans函数把它转成整形返回;即得到一个这样的函数:
[cpp]
int fun(){
return trans(gen());
}
int fun(){
return trans(gen());
}
我这里还是无参数的,所以不好使用lambda表达式。
最终方案与要点:
这个问题解决好久,心酸啊,过程就略过了,直接说最终方法。
假设使用前面的函数声明,那么这个函数对象应该这么生成:
[cpp]
boost::function<int()> fun=boost::bind(trans,boost::bind(gen));
boost::function<int()> fun=boost::bind(trans,boost::bind(gen));
其中要使用两次bind!
内层的那个看似没什么用,但是如果不适用它的话,boost不会以为我们要绑定的第一个参数是一个函数,而是以为我们要绑定一个具体的值,这个数字的值由这个gen函数产生。这时编译器会报“无法将double(void)类型转换为double类型”的错误。
同样的其他参数形式的(需要使用lambda库)、多层的函数嵌套绑定,都应该注意要把内层的函数通过bind变成一个可传递的对象,而不能直接写函数名!
bind绑定的各个东西都是对象(包括基本数据类型),是不支持把参数直接绑定为函数的,要这边做必须先把函数变成可以自由传递的对象!
另外不要以为你的参数已经是函数对象了就可以直接写了,还要再写bind(),具体原因还不清楚。
其他经验:
1,static_cast<T>()是运算符而不是函数!所以不能绑定。
2,boost/lambda/casts.hpp提供的ll_static_cast<T>虽然是函数,但是需要注意的是它是有两个模板参数的模板函数,在绑定的时候我们不能给出参数的实际类型(外侧的bind函数给ll_static_cast提供的是funnction<int()>的类型而不是int),因而也不能使用ll_static_cast,要自己写转换函数。
3,bind和lambda/bind有冲突,而且很多功能重叠,用一个就好了。
补充:软件开发 , C++ ,