3

这个问题如下:模板模板时强制特定重载

考虑以下代码:

#include <iostream>
#include <vector>
#include <array>
#include <type_traits>

// Version A
template<typename T>
void f(const T& x)
{
    std::cout<<"Version A"<<std::endl;
}

// Version B
template<typename... T1, template<typename...> class T>
void f(const T<T1...>& x)
{
    std::cout<<"Version B"<<std::endl;
}

// Version C
template<typename T1, typename TN, template<typename, TN...> class T, TN... N>
void f(const T<T1, N...>& x)
{
    std::cout<<"Version C"<<std::endl;
}

// Main
int main(int argc, char* argv[])
{
    f(double());
    f(std::vector<double>());
    f(std::array<double, 3>()); // <- How to force the use of Version C ?
    return 0;
}

默认情况下,它将产生(使用 GCC 4.7.1):

Version A
Version B
Version A

Version C当传递的类型是具有良好形状的模板时,如何强制使用(我可以添加新版本f,我可以添加std::enable_if或其他 C++11 类型特征语法,但如果可能的话,我想避免添加辅助类) ?

注意:这个技巧应该适用于每个整数类型 TN...

4

1 回答 1

3

不能在模板实参推导中推导出非类型模板实参的类型。这在标准中有明确规定:

14.8.2.5 从类型推导模板参数 [temp.deduct.type]

13 - 不能从非类型模板参数的类型推导出模板类型参数。
14 - [示例:

template<class T, T i> void f(double a[10][i]);
int v[10][20];
f(v); // error: argument for template-parameter T cannot be deduced

—结束示例]

这有效:

// Version C
template<typename T1, template<typename, size_t...> class T, size_t... N>
void f(const T<T1, N...>&) {
    std::cout<<"Version C"<<std::endl;
}
于 2012-12-17T10:49:51.307 回答