我正在处理一个事件库,而我正在面对Variadic模板的问题.
所有的工作非常好,除了我不能传递引用作为参数的事实…
这是一个非常简化的例子,用来揭示我的问题.
struct DelayedSignal
{
~DelayedSignal ()
{ std::cout << "~DelayedSignal CLOSE" << std::endl; }
template<class C,class... Args>
DelayedSignal ( void(C::*func)(Args...),C& obj )
{ std::cout << "DelayedSignal INIT - 03 - pointer to method & pointer to class instance (Arg num: " << sizeof...(Args) << ")" << std::endl; }
template<class C,C& obj,Args... args )
{
std::cout << "DelayedSignal INIT - 04 - pointer to method & pointer to class instance & arguments (Arg num: " << sizeof...(Args) << ")" << std::endl;
}
};
template<class... ArgsBis>
struct DelayedSignal_DebugHelper
{
~DelayedSignal_DebugHelper ()
{ std::cout << "~DelayedSignal_DebugHelper CLOSE" << std::endl; }
template<class C,class... Args>
DelayedSignal_DebugHelper ( void(C::*func)(Args...),C& obj )
{ std::cout << "DelayedSignal_DebugHelper INIT - 03 - pointer to method & pointer to class instance (Arg num: " << sizeof...(Args) << ")" << std::endl; }
template<class C,ArgsBis... args ) // Need to use ArgsBis instead of Args to make it work
{
std::cout << "DelayedSignal_DebugHelper INIT - 04 - pointer to method & pointer to class instance & arguments (Arg num: " << sizeof...(Args) << ")" << std::endl;
}
};
template < class Tr,class... Args >
struct Signal
{
void fire ( Args... args ) { std::cout << "Signal::fire::" << sizeof...(Args) << std::endl; }
};
struct Klass {};
int main()
{
std::string str1("Blop"); // Will be used as reference
Klass k; // Will be used as reference
Signal<void,Klass&> signal_01;
Signal<void,std::string&> signal_02;
std::cout << "====== DelayedSignal :: needed for production purpose ===============" << std::endl;
// OK
DelayedSignal test01(&Signal<void,std::string&>::fire,signal_02);
// HERE IS THE PROBLEM
//DelayedSignal test02(&Signal<void,signal_02,str1);
// OK
DelayedSignal test03(&Signal<void,Klass&>::fire,signal_01);
// HERE IS THE PROBLEM
//DelayedSignal test04(&Signal<void,signal_01,k);
std::cout << "====== DelayedSignal_DebugHelper :: used only for debug purpose ======" << std::endl;
// OK
DelayedSignal_DebugHelper<std::string&> test05(&Signal<void,signal_02);
// OK
DelayedSignal_DebugHelper<std::string&> test06(&Signal<void,str1);
// OK
DelayedSignal_DebugHelper<Klass&> test07(&Signal<void,signal_01);
// OK
DelayedSignal_DebugHelper<Klass&> test08(&Signal<void,k);
return 1;
}
当我将所有DelayedSignal实例注册到单个std :: list实例中时,我想避免在类本身使用模板,这就是为什么我在构造函数上使用模板.我也可以使用一个纯虚拟类作为所有DelayedSignal的基础,并将虚拟类的注册注册到std :: list中,但我认为最好最小化虚拟方法的使用,我真的很感兴趣这个问题?
正如你在这个例子中看到的那样,如果test02和test04被激活,返回错误.
DelayedSignal_DebugHelper与DelayedSignal几乎相同,除了在最后一个构造函数(而不是Args模板(方法模板参数))中使用ArgsBis(一个类模板参数),否则它不起作用(与DelayedSignal一样). Args被接受在void(C :: * func)(Args …)上,但不是与ArgsBis … args排除它们在同一个构造函数声明中的事实.
据我所知,只要没有参考(DelayedSignal test04(& Signal< void,Klass> :: fire,k)),或者有多个参数(或没有)引用.
有没有办法解决这个问题?
谢谢.
解决方法
Howard Hinnant是对的…另一种可能性是使用任何地方的引用,例如:
#include <iostream>
struct DelayedSignal
{
~DelayedSignal ()
{ std::cout << "~DelayedSignal CLOSE" << std::endl; }
template<class C,class... Args>
DelayedSignal ( void(C::*func)(Args &...),Args & ... args )
{
std::cout << "DelayedSignal INIT - 04 - pointer to method & pointer to class instance & arguments (Arg num: " << sizeof...(Args) << ")" << std::endl;
}
};
template < class Tr,class... Args >
struct Signal
{
void fire ( Args &... args ) { std::cout << "Signal::fire::" << sizeof...(Args) << std::endl; }
};
struct Klass {};
int main()
{
std::string str1("Blop"); // Will be used as reference
Klass k; // Will be used as reference
Signal<void,signal_02);
// HERE IS THE PROBLEM
DelayedSignal test02(&Signal<void,str1);
}