#include <iostream>
#include <boost/static_assert.hpp>
using namespace std;
// I understand how the following template function works
// template <class T>
// T GetMax (T a, T b) {
// T result;
// result = (a>b)? a : b;
// return (result);
// }
// I have difficulties to understand how the following code works
// when we should use this syntax
template<int i> void accepts_values_between_1_and_10() {
BOOST_STATIC_ASSERT(i >=1 && i < 10);
cout << "i(between 1 and 10): " << i << endl;
}
// I created the following function to test the understanding of BOOST_STATIC_ASSERT
template<typename T> void accepts_values_between_1_and_10_alternative(T i) {
BOOST_STATIC_ASSERT(i >=1 && i < 10);
cout << "i(between 1 and 10): " << i << endl;
}
int main () {
const int i = 5;
accepts_values_between_1_and_10<i>();
const int j = 6;
accepts_values_between_1_and_10_alternative(j);
return 0;
}
// $> g++ -o template_function -Wall -g template_function.cpp
// template_function.cpp: In function ‘void accepts_values_between_1_and_10_alternative(T) [with T = int]’:
// template_function.cpp:33:48: instantiated from here
// template_function.cpp:23:1: error: ‘((i > 0) && (i <= 9))’ is not a valid template argument for type ‘bool’ because it is a non-constant expression
问题1 &GT;以下语句背后的语法是什么?什么时候应该使用它?如果它是模板专业化,为什么我们必须提供参数而不仅仅是
template<int> void accepts_values_between_1_and_10()
instead of
template<int i> void accepts_values_between_1_and_10()
template<int> void accepts_values_between_1_and_10(int i)
instead of
template<int i> void accepts_values_between_1_and_10()
问题2 &GT;如果我们必须使用这种形式,那么我们必须在函数范围内采用这种语法吗?
问题3 &GT;如何更正accepts_values_between_1_and_10_alternative
的定义,以便它与BOOST_STATIC_ASSERT
一起使用?
谢谢
答案 0 :(得分:5)
您的template<typename T> void accepts_values_between_1_and_10_alternative(T i)
未编译,因为在实例化模板后评估运行时参数。在函数范围,语句BOOST_STATIC_ASSERT(i >=1 && i < 10);
表示正在查找i
作为非类型模板参数。但是,在封闭的命名空间范围内,i
是一个运行时参数,仅查找其类型以查找int
T
的实例化template<int i> void accepts_values_between_1_and_10()
。但是值6(即使它是程序中的常量表达式)仅在模板实例化之后进行计算,并且在模板参数推导期间在函数范围内不可见。
对于i
,正在查找5
的值以i
为BOOST_STATIC_ASSERT
实例化,此值会传回到{{1}}。