C++實(shí)現(xiàn)模板中的非類(lèi)型參數(shù)的方法
非類(lèi)型模板參看,顧名思義,模板參數(shù)不限定于類(lèi)型,普通值也可作為模板參數(shù)。在基于類(lèi)型的模板中,模板實(shí)例化時(shí)所依賴(lài)的是某一類(lèi)型的模板參數(shù),你定義了一些模板參數(shù)(template<typename T>)未加確定的代碼,直到模板被實(shí)例化這些參數(shù)細(xì)節(jié)才真正被確定。而非類(lèi)型模板參數(shù),面對(duì)的未加確定的參數(shù)細(xì)節(jié)是指(value),而非類(lèi)型。當(dāng)要使用基于值的模板時(shí),你必須顯式地指定這些值,模板方可被實(shí)例化。
在函數(shù)模板中使用非類(lèi)型參數(shù)
#include<iostream>
using namespace std;
//在函數(shù)模板中使用非類(lèi)型參數(shù)
template<class T>void Swap(T &a, T &b);
template<typename T, unsigned N>void Swap(T (&a)[N],T (&b)[N]);
template<typename T, unsigned N>void printArray(T (&arr)[N]);
int main(){
int m = 10, n = 90;
Swap(m,n);
cout << "m = " << m << ", n = " << n << endl;
int a[5] = { 1, 2, 3, 4, 5 };
int b[5] = { 10, 20, 30, 40, 50 };
Swap(a, b);
printArray(a);
printArray(b);
return 0;
}
template<class T> void Swap(T &a,T &b){
T temp = a;
a = b;
b = temp;
}
template<class T, unsigned N> void Swap(T (&a)[N],T (&b)[N]){
T temp;
for (int i = 0; i < N;i++){
temp = a[i];
a[i] = b[i];
b[i] = temp;
}
}
template<typename T, unsigned N>void printArray(T (&arr)[N]){
for (int i = 0; i < N;i++){
if (i == N-1){
cout << arr[i] << endl;
}
else{
cout << arr[i] << ", ";
}
}
}
在類(lèi)模板中使用非類(lèi)型參數(shù)
#include<iostream>
#include<cstring>
#include<cstdlib>
using namespace std;
//動(dòng)態(tài)數(shù)組實(shí)現(xiàn),在類(lèi)模板中使用非類(lèi)型參數(shù)
template<typename T,int N>
class Array{
public:
Array();
~Array();
public:
T & operator[](int i);//重載下標(biāo)運(yùn)算符
int length() const{ return m_length; }//獲取數(shù)組長(zhǎng)度
bool capacity(int n);//是否可改變數(shù)組容量
private:
int m_length;//數(shù)組當(dāng)前長(zhǎng)度
int m_capacity;//當(dāng)前內(nèi)存容量
T *m_p;//指向數(shù)組內(nèi)存的指針
};
template<typename T,int N>
Array<T, N>::Array(){
m_p = new T[N];
m_capacity = m_length = N;
}
template<typename T,int N>
Array<T, N>::~Array(){
delete[] m_p;
}
template<typename T,int N>
T & Array<T, N>::operator[](int i){
if (i<0||i>=m_length){
cout << "Exception:Array index out of bounds!" << endl;
}
return m_p[i];
}
template<typename T,int N>
bool Array<T, N>:: capacity(int n){
if (n>0){
int len = m_length + n;
if (len<=m_capacity){
m_length = len;
return true;
}
else{
T *pTemp = new T[m_length + 2 * n*sizeof(T)];
if (NULL==pTemp){
cout << "Exception: Failed to allocate memory!";
return false;
}
else{
memcpy(pTemp,m_p,m_length*sizeof(T));
delete[] m_p;
m_p = pTemp;
m_capacity = m_length = len;
}
}
}
else{
int len = m_length - abs(n);
if (len<0){
cout << "Exception:Array length is too small!" << endl;
return false;
}
else{
m_length = len;
return true;
}
}
}
int main(){
Array<int, 5> arr;
for (int i = 0, len = arr.length(); i < len;i++){
arr[i] = 2 * i;
}
cout << "first print:" << endl;
for (int i = 0, len = arr.length(); i < len;i++){
cout << arr[i] << " ";
}
cout << endl;
//擴(kuò)大容量為增加的元素賦值
arr.capacity(8);
for (int i = 5, len = arr.length(); i < len;i++){
arr[i] = 2 * i;
}
cout << endl;
cout << "second print:" << endl;
for (int i = 0, len = arr.length(); i < len;i++){
cout << arr[i] << " ";
}
cout << endl;
arr.capacity(-4);
cout << "third print: " << endl;
for (int i = 0, len = arr.length(); i < len; i++){
cout << arr[i] << " ";
}
cout << endl;
return 0;
}
非類(lèi)型模板參數(shù)的限制
非類(lèi)型模板參數(shù)是有類(lèi)型限制的。一般而言,它可以是常整數(shù)(包括enum枚舉類(lèi)型)或者指向外部鏈接對(duì)象的指針。
浮點(diǎn)數(shù)和類(lèi)對(duì)象(class-type)不允許作為非類(lèi)型模板參數(shù):
template<double VAL> // ERROR: 浮點(diǎn)數(shù)不可作為非類(lèi)型模板參數(shù)
double process(double v)
{
return v * VAL;
}
template<std::string name> // ERROR:類(lèi)對(duì)象不能作為非類(lèi)型模板參數(shù)
class MyClass
{}
稍作變通,我們即可使編譯通過(guò):
template<double* PVAL>
double process(const double& x)
{
return x * (*PVAL);
}
template<const char* name>
class MyClass
{
...
}
這樣可順利通過(guò)編譯,但如果想在當(dāng)前文件中使用這兩個(gè)模板,還需要?jiǎng)右恍┦帜_:
double val = 10; double res = process<&val>(20); // ERROR: 表達(dá)式必須含有常量值 MyClass<"hello"> x; // ERROR: 模板參數(shù)不能引用非外部實(shí)體 const char* s = "hello"; MyClass<s> x; // ERROR: 表達(dá)式必須含有常量值
這里就點(diǎn)出另外一點(diǎn)注意事項(xiàng),也就是非類(lèi)型模板參數(shù)的限制,非類(lèi)型模板參數(shù)可以是指針,但該指針必須指向外部鏈接對(duì)象,還記得在A.cpp中如何引用B.cpp中的全局變量嗎,在A.hpp中使用extern關(guān)鍵字對(duì)外部變量加以引用。
// B.cpp double val = 3.14159265; char str[] = "hello";
// A.hpp extern double val; extern char str[];
// A.cpp #include "A.hpp" double res = process<&val>(10); MyClass<str> x;
到此這篇關(guān)于C++實(shí)現(xiàn)模板中的非類(lèi)型參數(shù)的方法的文章就介紹到這了,更多相關(guān)C++ 模板非類(lèi)型參數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
如何寫(xiě)好C main函數(shù)的幾個(gè)注意事項(xiàng)
這篇文章主要介紹了如何寫(xiě)好C main函數(shù)的幾個(gè)注意事項(xiàng),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06
C++?Boost?ProgramOptions超詳細(xì)講解
Boost是為C++語(yǔ)言標(biāo)準(zhǔn)庫(kù)提供擴(kuò)展的一些C++程序庫(kù)的總稱(chēng)。Boost庫(kù)是一個(gè)可移植、提供源代碼的C++庫(kù),作為標(biāo)準(zhǔn)庫(kù)的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開(kāi)發(fā)引擎之一,是為C++語(yǔ)言標(biāo)準(zhǔn)庫(kù)提供擴(kuò)展的一些C++程序庫(kù)的總稱(chēng)2022-11-11
C++?IO設(shè)備讀寫(xiě)功能實(shí)現(xiàn)詳解
C++的文件IO(Input,Output)操作就是指對(duì)文件進(jìn)行讀寫(xiě)(輸入與輸出)的操作。輸入就是從磁盤(pán)上的文件中讀取內(nèi)容到內(nèi)存中。輸出就是將內(nèi)存中的數(shù)據(jù)內(nèi)容輸出或者說(shuō)寫(xiě)入到磁盤(pán)的文件中,這篇文章主要介紹了C++?IO設(shè)備讀寫(xiě)功能實(shí)現(xiàn)2022-11-11
C語(yǔ)言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)快速排序圖文示例
這篇文章主要為大家介紹了C語(yǔ)言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)快速排序圖文示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05

