我得到一个“错误的模板参数数量(2,应该是1)”错误,我无法理解 .
我有一个类为其他想要与之交互的类型提供一些辅助函数,设置第一个模板参数,在创建时它们必须自动相互兼容 . 为了以方便,通用的方式执行此操作,我决定使用可变参数模板,该模板旨在传递构造函数参数和要创建的对象类型的其他模板参数:
template<typename INTERNAL_TYPE>
class Linker
{
template< template<typename, typename ...> class INPUT_OBJ_TYPE, class ... TEMPLATE_ARGS, class ... CONSTRUCTOR_ARGS >
std::shared_ptr< INPUT_OBJ_TYPE<INTERNAL_TYPE,TEMPLATE_ARGS ...> > getLinked( CONSTRUCTOR_ARGS ... args )
{
std::shared_ptr< INPUT_OBJ_TYPE<INTERNAL_TYPE,TEMPLATE_ARGS ...> > ptr = std::make_shared< INPUT_OBJ_TYPE<INTERNAL_TYPE,TEMPLATE_ARGS ...> >( args... );
return ptr;
}
};
这适用于以下类:
template<typename INTERNAL_TYPE, typename SECOND_TYPE>
class TEST_CLASS_1
{};
也就是说,我可以做到以下几点:
Linker<int> container;
auto test_1 = container.getLinked<TEST_CLASS_1,double>();
然后我尝试对另一个只在一个参数上进行模板化的类做同样的事情:
template<typename INTERNAL_TYPE>
class TEST_CLASS_2
{};
auto test_2 = container.getLinked<TEST_CLASS_2>();
但得到上面提到的错误..为什么?如果我从函数声明/定义中删除了TEMPLATE_ARGS,那么我可以使用代码编写第二个测试(尽管第一个测试不再编译) . 所以我认为那时的编译器还没有意识到TEMPLATE_ARGS对于第二次测试是空的,并且因为有太多的模板参数而引发错误 . 所以我想我可能需要使用跟踪返回类型
template< template<typename, typename ...> class INPUT_OBJ_TYPE,
class ... TEMPLATE_ARGS, class ... CONSTRUCTOR_ARGS >
auto getLinked( CONSTRUCTOR_ARGS ... args )
-> std::shared_ptr< INPUT_OBJ_TYPE<TREE_TYPE,TEMPLATE_ARGS ...> >
或使用 decltype
,但这也不起作用:
template< template<typename, typename ...> class INPUT_OBJ_TYPE,
class ... TEMPLATE_ARGS, class ... CONSTRUCTOR_ARGS >
auto getLinked( CONSTRUCTOR_ARGS ... args )
->decltype(std::shared_ptr< INPUT_OBJ_TYPE<TREE_TYPE,TEMPLATE_ARGS ...> >())
我对这个问题是对的吗?我该如何解决?
非常感谢!
2 回答
这可能是编译器错误 .
获得所需内容的方法是提供两个函数,原始函数和另一个不具有不可推导参数包的函数:
似乎
TEST_CLASS_2
的初始替换正在命中编译器的"every valid specialization requires an empty pack"代码路径 - 也就是说,以下模板格式错误,无需诊断:因为它的每个有效特化都要求
TArgs
是一个空包 . 最初替换为getLinked
确实会产生类似的东西,但当然,你并没有确信应该有错误,但无论如何 .一种可能的解决方法是推迟替换
INPUT_OBJ_TYPE
直到扣除后,通过采用标记类型并推导出所有内容: