универсальный указатель на функцию-член в качестве параметра шаблона - PullRequest
21 голосов
/ 20 марта 2012

Рассмотрим этот код:

#include <iostream>
using namespace std;

class hello{
public:
    void f(){
        cout<<"f"<<endl;
    }
    virtual void ff(){
        cout<<"ff"<<endl;
    }
};

#define call_mem_fn(object, ptr)  ((object).*(ptr))

template<R (C::*ptr_to_mem)(Args...)> void proxycall(C& obj){
    cout<<"hello"<<endl;
    call_mem_fn(obj, ptr_to_mem)();
}

int main(){
    hello obj;
    proxycall<&hello::f>(obj);
}

Конечно, это не скомпилируется в строке 16, потому что компилятор не знает, что такое R, C и Args. Но есть другая проблема: если кто-то пытается определить эти параметры шаблона прямо перед ptr_to_mem, он сталкивается с этой плохой ситуацией:

template<typename R, typename C, typename... Args, R (C::*ptr_to_mem)(Args...)> 
                             //  ^variadic template, but not as last parameter!
void proxycall(C& obj){
    cout<<"hello"<<endl;
    call_mem_fn(obj, ptr_to_mem)();
}

int main(){
    hello obj;
    proxycall<void, hello, &hello::f>(obj);
}

Удивительно, но g ++ не жалуется на то, что Args не является последним параметром в списке шаблонов, но в любом случае он не может связать proxycall с нужной функцией шаблона, и просто отмечает, что это возможный кандидат.

Есть решение? Мое последнее средство - передать указатель на функцию-член в качестве аргумента, но если бы я мог передать его в качестве параметра шаблона, он бы лучше подходил к остальной части моего кода.

EDIT: как указывали некоторые, пример кажется бессмысленным, потому что proxycall не собирается передавать аргументы. Это не так в реальном коде, над которым я работаю: аргументы извлекаются с некоторыми шаблонными приемами из стека Lua. Но эта часть кода не имеет отношения к вопросу и довольно длинна, поэтому я не буду вставлять ее сюда.

1 Ответ

36 голосов
/ 20 марта 2012

Вы можете попробовать что-то вроде этого:

template <typename T, typename R, typename ...Args>
R proxycall(T & obj, R (T::*mf)(Args...), Args &&... args)
{
    return (obj.*mf)(std::forward<Args>(args)...);
}

Использование: proxycall(obj, &hello::f);

В качестве альтернативы, чтобы превратить PTMF в аргумент шаблона, попробуйте специализацию:

template <typename T, T> struct proxy;

template <typename T, typename R, typename ...Args, R (T::*mf)(Args...)>
struct proxy<R (T::*)(Args...), mf>
{
    static R call(T & obj, Args &&... args)
    {
        return (obj.*mf)(std::forward<Args>(args)...);
    }
};

Использование:

hello obj;

proxy<void(hello::*)(), &hello::f>::call(obj);

// or

typedef proxy<void(hello::*)(), &hello::f> hello_proxy;
hello_proxy::call(obj);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...