用于重载的成员函数指针

时间:2016-12-05 19:06:18

标签: c++ casting pointer-to-member

我希望有一个更好的"指向方法的指针的转换操作。

给定一个具有重载成员函数的类和一个模板,该模板可以使用各种成员指针和对象并简单地调用它们。

class A
{
    public:
        void Do( int i ) { std::cout << "Hallo with int " << i << std::endl;}
        void Do( float f ) { std::cout << "and with float " << f << std::endl;}
        void Do( std::string s ) { std::cout << "also with string " << s << std::endl;}
};

    template <typename OBJECT, typename METHOD, typename ... PARMS>
void Call( OBJECT& obj, METHOD m, PARMS ... parms)
{
    std::cout << __PRETTY_FUNCTION__ << std::endl;

    (obj.*m)(parms...);
}

现在,如果我想使用Call模板,我必须手动转换指针以获取正确版本的重载函数,如:

   Call( a, static_cast< void(A::*)(int)>( &A::Do ), 1);
   Call( a, static_cast< void(A::*)(float)>( &A::Do ), 8.88);
   Call( a, static_cast< void(A::*)(std::string)>( &A::Do ), "Willi");

所以我喜欢让演员更容易阅读并得到这个助手类:

template <typename > class Cast;

template < typename RET, typename ... ARGS >
class Cast< RET(ARGS...)>
{
    public:
        template < typename OBJ >
            static auto From( RET(OBJ::*ptr)(ARGS...))-> RET(OBJ::*)(ARGS...)
            {
                return ptr;
            }
};

并可以将其用作:

    Call( a, Cast<void(int)>::From( &A::Do ), 100);
    Call( a, Cast<void(float)>::From( &A::Do ), 4.46);
    Call( a, Cast<void(std::string)>::From( &A::Do ),"Check it");

问: 是否可以使我的类Cast成为模板函数,以便更轻松地获取转换调用的语法?我想要一些可以像这样调用的东西:

Call( a, Cast<void(std::string)>(  &A::Do ), "Yeeeaaa!");

但是我找不到以这种方式编写模板函数的方法,因为我写的任何一种特殊化都是不允许的。

允许使用任何类型的c ++。如果C ++ 17提供了一种新方法,请告诉我!

为什么我不能在调用函数中使用lambda的参数:

我的真正代码只需要注册指针但不要直接调用指针。因此,演员函数也必须使用:

Register( a, Cast<void(int)>::From( &A::Do ));

1 个答案:

答案 0 :(得分:2)

您不必将参数和结果类型分开,只是为了再次加入它们。只需声明一个指向函数类型的成员指针:

template <typename F, typename C>
auto Cast(F (C::*ptr)) {
    return ptr;    
}

您想要的语法有效:

Call(a, Cast<void(std::string)>(&A::Do), "Yeeeaaa!");