C++模板特化:类型与常量的灵活掌控

一、模板参数再介绍

初级模板知识

模板参数是一个用来存放类型名称(int double

等内置类型和自定义类型名称)的变量。在代码实现中使用模板参数写代码(写一个函数或类),会增加代码复用的能力。

写出的函数或类被称为函数模板和类模板。 模板实例化:编译器根据使用模板时指定的类型名称,用这个类型名称来替代模板参数,生成相应的函数或类。

使用函数模板时,通过传入参数类型实例化出相应的模板。 而使用类模板时,需要显示说明模板参数类型,才能实例化出相应的类。

二、非类型模板参数

前面提及的模板参数接受的都是类型的名字,还有一种模板参数接受的是一个常数。

看下面一段代码:

cpp 复制代码
#include<iostream>
template<class T,size_t size=10>
class arr {
private:
	T arr[size];
	int len = size;
public:
	arr();
	~arr();
	T& operator [](const arr& a);
	bool empty();
};

这个是C++STL中对数组的改造array的模拟实现(未完成),与c风格数组相比,容器arr对越界访问更加严格。我们看到arr是一个类模板,有两个模板参数,第二个模板参数就是上面提及的非类型模板参数,相当于C语言中常量的宏定义。

但与宏定义相比较,这样的类模板可以根据传入参数的不同自动调整常量,不需要改变代码,但是宏定义恰恰相反。

举个arr的使用例子来体会非类型模板参数的用途:

cpp 复制代码
#include<iostream>
using namespace std;
#include<array>
int main()
{
	array<int, 20> arr = { 1,2,3,4,5 };
	for (int i = 0; i < arr.size(); i++)
	{
		cout<<arr[i]<<" ";//1 2 3 4 5 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
	}
	cout << endl;
	return 0;
 }

三、函数模板的特化

为什么需要引入特化

就函数模板而言:由于所有符合函数模板的参数列表中的每组参数并不都要根据函数模板实例化出的函数进行运算,总有一两个特例要实现不同的方法,所以引入模板特化。比如我想实现一个函数,当两个参数是一切参数时,判断第一个参数是否大于第二个参数。如果这样实现:

cpp 复制代码
#include<iostream>
using namespace std;
template<class T>
bool fun(T a, T b)
{
	return a>b

}

int main()
{
	int a = 10, b = 20;
	int* pb = &b, * pa = &a;
	cout << fun(a, b) << endl;
	cout << fun(pa, pb) << endl;
	
}

存在指针比较,是不是没有任何意义,当参数为int*时,我们特化一个函数模板,让它返回指针所指向的最大的数:

cpp 复制代码
#include<iostream>
using namespace std;
template<class T>
bool fun(T a, T b)
{
	return a>b

}
template<>
bool fun<int*>(int* a, int* b)
{
	return *a > *b;
}
int main()
{
	int a = 10, b = 20;
	int* pb = &b, * pa = &a;
	cout << fun(a, b) << endl;
	cout << fun(pa, pb) << endl;
	
}

函数模板特化的语法:

1.前提是有一个相关的函数模板

2.在特化函数模板之前加上template<>

3.在特化声明函数时在函数名之后参数表之前加上<特化的类型名称>

4.要求函数的其余部分除将模板参数特化替代外,必须和原来函数一模一样否则,就会编译报错。

函数模板特化的特点:

由于函数模板的性质,当定义一个将函数模板中所有模板参数替代成统一类型名称的函数fun时,调用一个既可以走函数模板,又可以直接调用fun时,编译器会不再实例化函数模板,直接调用fun。因此函数模板特化的作用不大,大多数情况都可以通过定义一个将函数模板中所有模板参数替代成统一类型名称的函数fun,调用fun直接解决问题,但也不排除只有使用函数模板特化的方法才能解决问题的项目。

使用上面提及的第二种常规方法解决之前需要函数模板特化才可以解决的问题:

cpp 复制代码
#include<iostream>
using namespace std;
template<class T>
bool fun(T a, T b)
{
	return a > b;

}

bool fun(int* a, int* b)
{
	return *a > *b;
}
int main()
{
	int a = 10, b = 20;
	int* pb = &b, * pa = &a;
	cout << fun(a, b) << endl;
	cout << fun(pa, pb) << endl;
	
}

四、类模板的特化

1.全特化

顾名思义就是将所有参数都特化,和函数模板特化一样,这里也强调一下:函数模板特化的方式只有一种,类模板特化可以全特化,也可以半特化。

示例代码:

cpp 复制代码
#include<iostream>
using namespace std;
template<class T1, class T2>
class c1
{private:
	T1 t1;
	T2 t2;
public:
	c1()
	{
		cout << "T1,T2" << endl;
	}
};
template<>
class c1<int,int>
{
private:
	int  t1;
	int t2;
public:
	c1()
	{
		cout << "int ,int " << endl;
	}
};
int main()
{
	c1<int, double> cc;
	c1<int, int> c1;
	//T1,T2
	//int, int

}

和函数模板特化一致,特化定义前要加上template<>

特化声明时l类的名字和{}之间要加上<特化的类型名称>

2.半特化(别称:偏特化)

先来看一下半特化的代码演示:

cpp 复制代码
#include<iostream>
using namespace std;
template<class T1, class T2>
class c1
{private:
	T1 t1;
	T2 t2;
public:
	c1()
	{
		cout << "T1,T2" << endl;
	}
};
template<class T1>
class c1<T1,int>
{
private:
	T1  t1;
	int t2;
public:
	c1()
	{
		cout << "T1 ,int " << endl;
	}
};
int main()
{
	c1<int, double> cc;
	c1<int, int> c1;
	//T1,T2
	//T1, int

}

半特化和全特化的不同在于:半特化定义之前,加的是template<未特化的模板参数声明>特化时类名称和{}之间加的是<未特化的模板参数,特化的模板参数>(这个排序要求和类模板定义之前的模板参数声明要一一对应下面的例子就是最好的说明)

cpp 复制代码
#include<iostream>
using namespace std;
template<class T1, class T2,class T3>
class c1
{private:
	T1 t1;
	T2 t2;
	T3 t3;
public:
	c1()
	{
		cout << "T1,T2,T3" << endl;
	}
};
template<class T1,class T3>
class c1<T1,int,T3>
{
private:
	T1  t1;
	int t2;
public:
	c1()
	{
		cout << "T1 ,int,T3 " << endl;
	}
};
int main()
{
	c1<int, double,int> cc;
	c1<int, int,double> c1;
	/*T1, T2, T3
		T1, int, T3*/

}

半特化还有一种特殊的例子,当被特化的类未指针或引用时:

cpp 复制代码
template<class T1, class T2>
class Data
{
public:
Data() {cout<<"Data<T1, T2>" <<endl;}
private:
T1 _d1;
T2 _d2;
};//两个参数偏特化为指针类型
template <typename T1, typename T2>
class Data <T1*, T2*>
{
public:
Data() {cout<<"Data<T1*, T2*>" <<endl;}
private:
T1 _d1;
T2 _d2;
};
//两个参数偏特化为引用类型
template <typename T1, typename T2>
class Data <T1&, T2&>
{
public:
Data(const T1& d1, const T2& d2)
: _d1(d1)
, _d2(d2)
{
cout<<"Data<T1&, T2&>" <<endl;
}
private:
const T1 & _d1;
const T2 & _d2;
};
void test2 ()
{
Data<int , double> d2; // 调用基础的模板
Data<int *, int*> d3; // 调用特化的指针版本
Data<int&, int&> d4(1, 2); // 调用特化的指针版本
}
相关推荐
XMYX-0几秒前
40 - Go HTTP 客户端:从 http.Get 到高性能连接池
开发语言·http·golang
Daydream.V2 分钟前
C++ 入门全攻略:从基础语法到核心特性
java·开发语言·c++
我是一颗柠檬3 分钟前
【JDK8新特性】接口默认方法与静态方法Day8
java·开发语言·后端·intellij-idea
lulu12165440787 分钟前
【开发者指南】Gemini 3.5开发入门:从API调用到Agent构建
java·开发语言·人工智能·python·ai编程
盲敲代码的阿豪8 分钟前
Python 爬虫入门基础教程:从入门到实践
开发语言·爬虫·python
我能坚持多久11 分钟前
STL详解——stack以及queue的模拟实现
开发语言·c++·学习
无限进步_12 分钟前
【C++】智能指针的设计逻辑:RAII与资源安全
c++·算法·安全
会周易的程序员13 分钟前
AI 编程助手:从“猫弄乱的线团”到“击鼓传花”的 Bug 修复
c++·人工智能·物联网·架构·bug·iot
江屿风20 分钟前
C++OJ题经验总结(竞赛)2
开发语言·c++·笔记·算法