我有一个C ++类,我正在尝试使用std :: bind1st将成员函数绑定到'this'参数。例如:
class MyClass
{
public:
void Foo()
{
using namespace std;
// this works fine
this->Bar();
// this also works fine
mem_fun( &MyClass::Bar )( this );
// this does not
bind1st( mem_fun( &MyClass::Bar ), this )();
// this is not a possibility for this program
boost::bind( &MyClass::Bar, this )();
};
void Bar()
{
};
};
当我添加最后一个'bind1st'行时,我得到以下编译器错误:
1>stl/_function.h(189) : error C2039: 'second_argument_type' : is not a member of 'stlp_std::mem_fun_t<_Ret,_Tp>'
1> with
1> [
1> _Ret=void,
1> _Tp=MyClass
1> ]
1> .\MyClass.cpp(50) : see reference to class template instantiation 'stlp_std::binder1st<_Operation>' being compiled
1> with
1> [
1> _Operation=stlp_std::mem_fun_t<void,MyClass>
1> ]
1>stl/_function.h(189) : error C2146: syntax error : missing ',' before identifier 'second_argument_type'
1>stl/_function.h(189) : error C2065: 'second_argument_type' : undeclared identifier
1>stl/_function.h(190) : error C2955: 'stlp_std::unary_function' : use of class template requires template argument list
1> stl/_function_base.h(40) : see declaration of 'stlp_std::unary_function'
1>stl/_function.h(191) : error C2039: 'second_argument_type' : is not a member of 'stlp_std::mem_fun_t<_Ret,_Tp>'
1> with
1> [
1> _Ret=void,
1> _Tp=MyClass
1> ]
1>stl/_function.h(191) : error C2146: syntax error : missing ',' before identifier 'second_argument_type'
1>stl/_function.h(191) : error C2065: 'second_argument_type' : undeclared identifier
1>stl/_function.h(194) : error C2955: 'stlp_std::unary_function' : use of class template requires template argument list
1> stl/_function_base.h(40) : see declaration of 'stlp_std::unary_function'
1>stl/_function.h(195) : error C2955: 'stlp_std::unary_function' : use of class template requires template argument list
1> stl/_function_base.h(40) : see declaration of 'stlp_std::unary_function'
1>stl/_function.h(197) : error C2146: syntax error : missing ';' before identifier '_ArgParamType'
1>stl/_function.h(197) : error C3254: 'stlp_std::binder1st<_Operation>' : class contains explicit override 'param_type' but does not derive from an interface that contains the function declaration
1> with
1> [
1> _Operation=stlp_std::mem_fun_t<void,MyClass>
1> ]
1>stl/_function.h(197) : error C2838: 'param_type' : illegal qualified name in member declaration
1>stl/_function.h(197) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
1>stl/_function.h(197) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
1>stl/_function.h(198) : error C2146: syntax error : missing ';' before identifier '_ConstArgParamType'
1>stl/_function.h(198) : error C3254: 'stlp_std::binder1st<_Operation>' : class contains explicit override 'const_param_type' but does not derive from an interface that contains the function declaration
1> with
1> [
1> _Operation=stlp_std::mem_fun_t<void,MyClass>
1> ]
1>stl/_function.h(198) : error C2838: 'const_param_type' : illegal qualified name in member declaration
1>stl/_function.h(198) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
1>stl/_function.h(198) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
1>stl/_function.h(199) : error C2039: 'first_argument_type' : is not a member of 'stlp_std::mem_fun_t<_Ret,_Tp>'
1> with
1> [
1> _Ret=void,
1> _Tp=MyClass
1> ]
1>stl/_function.h(199) : error C2146: syntax error : missing ',' before identifier 'first_argument_type'
1>stl/_function.h(199) : error C2065: 'first_argument_type' : undeclared identifier
1>stl/_function.h(199) : error C2955: 'stlp_std::__call_traits' : use of class template requires template argument list
1> stl/type_traits.h(452) : see declaration of 'stlp_std::__call_traits'
1>stl/_function.h(203) : error C2039: 'first_argument_type' : is not a member of 'stlp_std::mem_fun_t<_Ret,_Tp>'
1> with
1> [
1> _Ret=void,
1> _Tp=MyClass
1> ]
1>stl/_function.h(203) : error C2146: syntax error : missing ';' before identifier '_M_value'
1>stl/_function.h(203) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
1>stl/_function.h(203) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
1>stl/_function.h(208) : error C2061: syntax error : identifier '_ConstArgParamType'
1>stl/_function.h(211) : error C2061: syntax error : identifier '_ArgParamType'
1>stl/_function.h(212) : error C2535: '_Result stlp_std::binder1st<_Operation>::operator ()(void) const' : member function already defined or declared
1> with
1> [
1> _Operation=stlp_std::mem_fun_t<void,MyClass>
1> ]
1> stl/_function.h(208) : see declaration of 'stlp_std::binder1st<_Operation>::operator ()'
1> with
1> [
1> _Operation=stlp_std::mem_fun_t<void,MyClass>
1> ]
我正在使用STLPort v5.2.1进行标准库实现。
通常,我会切换到boost :: bind并使用它。不幸的是,这不适用于此应用程序。
我可以做些什么来获得我正在寻找的功能?
谢谢, PaulH
编辑:
更清楚的是,我正在寻找一种方法来使一元函数适应你所谓的不带参数的函数。我想将this
'绑定到MyClass::Bar
。
class SomeOtherClass
{
public:
template< typename Fcn >
void ExecuteFcn( Fcn fcn )
{
fcn();
};
};
some_other_class.ExecuteFcn( bind1st( mem_fun( &MyClass::Bar ), this );
答案 0 :(得分:5)
你想用bind1st做什么?
bind1st
函数采用二进制函数,并通过隐含第一个参数使其适应一元函数(这可能不是最好的描述,对不起)。 mem_fun
的返回值是一元函数。
mem_fun
函数返回成员函数的适配器。改编后的函数不接受任何参数,但返回的适配器接受一个参数,一个指向要使用的MyClass对象的指针。
因此,基本上,您的mem_fun
调用返回一个接受一个参数的适配器,但是bind1st
期望一个带有两个参数的适配器。有一个mem_fun1
返回一个带有两个参数的适配器,第一个是对象指针,第二个是函数的参数,但这不是你想要的。
事实上,我不太明白你要做什么;普通mem_fun
版本为何不合适?如果您需要将对象指针“附加”到适配器,我认为您不能使用当前标准库执行此操作,除非您使用bind。
当然,您可以创建一个包含对象指针的包装类,然后定义operator()以使用该对象调用适配器。
// Quick and dirty example of this.
// You could extend this with a second template parameter for return type, if
// needed. Just be sure to specialize for void if you do that.
template<typename Object>
struct bound_mem_fun_t {
bound_mem_fun_t(mem_fun_t<void, Object> f, Object* o) : fun(f), obj(o) { }
void operator()() { fun(obj); }
mem_fun_t<void, Object> fun;
Object* obj;
};
MyClass a;
bound_mem_fun_t<MyClass> func(mem_fun(&MyClass::Bar), &a);
func();
答案 1 :(得分:4)
bind1st
需要二进制函数,并返回一元函子。在这里,您传递一元函数。
“绑定”的目的是通过将其中一个参数设置为给定值来使用二元函数/仿函数(如带有一个参数的非静态方法),就像一元参数一样。
这可行(注意Bar
的新原型):
class MyClass
{
public:
void Foo()
{
using namespace std;
// this works fine
bind1st( mem_fun( &MyClass::Bar ), this )(42);
};
void Bar(int i)
{
};
};
答案 2 :(得分:1)
我想为此添加另一个可能的解决方案,那就是使用C ++ 11中的lambda函数。
这是一个简单的例子:
class MyClass
{
std::string mName;
public:
MyClass(const char* name) : mName(name) {};
void Foo()
{
std::cout << "My name is " << mName << std::endl;
};
};
void CallMyFn(std::function<void()> fn)
{
fn();
}
int main()
{
MyClass myInstance("Ishmael");
CallMyFn( [&]() { myInstance.Foo(); } );
return 0;
}
输出:“我的名字是以实玛利。”
答案 3 :(得分:0)
我认为它不起作用,因为bind1st()
需要二进制函数对象并返回一元函数对象。由于Bar()
不带参数,mem_fun()
正在为您提供一元函数对象。 bind1st()
不知道如何处理。
你如何打算使用mem_fun()
生成的函数对象?