轉載地址:http://www.cppblog.com/Tim/archive/2012/11/09/194965.html
我們知道,用模板的時候可以爲他提供兩種類型的模板形參:一種是類型模板形參,一種是非類型模板形參。例如如下聲明:
template<class T,int n>void foo(T t); //用了類型模板形參,同時還用到了非類型模板形參int
兩者的區別是:對應類型模板形參,編譯器會根據實參(對應模板函數)或者用戶指定類型來實例化對應的模板函數或類型。而非類型模板形參主要是用來在模板函數調用時指定該形參的值。
非類型模板形參最常用的是用來自動獲取數組的維數。比如,我們給某個函數傳遞一個數組,並在函數內部打印這個數組的值,如果沒有模板,我們的做法就可能是如下這樣:
{
for (int i=0;i<N;++i)
{
cout<<r[i]<<",";
}
cout<<endl;
}
但是,如果是用非類型模板參數,又該如何用呢?
我們如果給出如下的方案,並不能完全達到我們的目的:
void print(T r[N])
{
for (int i=0;i<N;++i)
{
cout<<r[i]<<",";
}
cout<<endl;
}
int ss[] = {1,2,3,4,5,6,7,8,9};
print(ss); //或者print<int>(ss);
return 0;
}
其實,這完全不能怪編譯器或者模板,因爲只要這樣寫就完全可以了:
void print(T (&r)[N])
{
for (int i=0;i<N;++i)
{
cout<<r[i]<<",";
}
cout<<endl;
}
其實原理是很簡單的,如果我們直接將數組名傳給一個函數,數組將退化爲一個指針,所以編譯器自然也就無法用指針來推導維數了。但是我們強調傳的是引用的話,編譯器就明確知道這是一個數組,所以就去推導維數。我們可以給上面的程序加一句打印語句來測試:
void print(T (&r)[N])
{
cout<<sizeof(r)<<endl;
for (int i=0;i<N;++i)
{
cout<<r[i]<<",";
}
cout<<endl;
}
template<class T,int N>
void print1(T r[N])
{
cout<<sizeof(r)<<endl;
for (int i=0;i<N;++i)
{
cout<<r[i]<<",";
}
cout<<endl;
}
int main()
{
int ss[] = {1,2,3,4,5,6,7,8,9};
print(ss);
print1<int,9>(ss);
return 0;
}
36
1,2,3,4,5,6,7,8,9,
4
1,2,3,4,5,6,7,8,9,
很明顯可以看出,傳引用的方式他知道是一個數組,所以sizeof是按數組的方式計算的4*9=36.而第二種方式儘管我告訴他維數了,並且打印出結果了,但是絲毫沒有改變編譯器將print1的參數T r[N]視爲一個指針的事實,所以sizeof的值爲4.
好了,總結一下就是:如果希望用非類型模板形參的方式取數組的維數時,請記住用數組引用的方式作爲形參!