【ONE·C++ || 模板进阶】

在这里插入图片描述

总言

  主要介绍模板相关内容:非类型模板参数、类模板特化、模板的分离编译。
  


  
  
  
  
  
  

1、非类型模板参数

  基本概念:
  🎯你能解释一下什么是非类型模板参数吗?它与类型模板参数有何不同?
  🎯非类型模板参数在C++模板编程中扮演了什么角色?

  使用场景:
  🎯你能给出一些使用非类型模板参数的典型场景或例子吗?
  🎯为什么在这些场景中,使用非类型模板参数是合适的?

  参数类型:
  🎯非类型模板参数可以是什么类型的数据?
  🎯为什么不能使用所有类型的数据作为非类型模板参数?

  
  
  
  

1.1、主要介绍

  1)、问题引入
  在之前(模板初阶),我们已经对模板有一定了解:以下为一个类模板。

#define N 5
template<class T>
class Array
{
private:
	T _a[N];
};

int main()
{
	Array<int>  a1;
	Array<double> a2;

	return 0;
}

在这里插入图片描述
  根据上述情况,我们能使用模板定义出两个类型不同的类,但是,假如我们需要a1大小为10,a2大小为8,该如何定义呢?
  
  这时我们就需要非类型模板参数
  
  
  2)、非类型模板参数介绍
  模板参数可以分为类类型形参非类型形参
  类型形参:出现在模板参数列表中,跟在class或者typename之后的参数类型名称。
  非类型形参:用一个常量作为类(函数)模板的一个参数,在类(函数)模板中可将该参数当成常量来使用。
  
  举例如下:此处的size_t N即非类型模板参数。

template<class T, size_t N>
class Array
{
private:
	T _a[N];
};


int main()
{
	Array<int,10>  a1;
	Array<double,8> a2;

	return 0;
}

在这里插入图片描述

  特点介绍:
  1、非类型模板参数只能是常量,因此其限制了变长数组的使用。(即,非类型模板参数编译时必须是常量表达式,这意味着参数的值在编译时必须是已知的,且不能依赖于运行时才能确定的值。)
  2、非类型模板参数也可以使用缺省值

template<class T, size_t N=5>
struct Array
{
	T _a[N];
};

int main()
{
	Array<int>  a1;
	Array<double,8> a2;

	return 0;
}

在这里插入图片描述

  3、非类型模板参数的类型必须足够简单,以便编译器能够在编译时确定其值。这通常意味着它们必须是整型(包括intcharenum等)、指针类型(包括指针之间的差值)、或者 std::nullptr_t。其他复杂类型(如类类型、浮点数、数组等)不能作为非类型模板参数。
在这里插入图片描述

  4、非类型的模板参数必须在编译期就能确认结果
  
  
  
  
  
  

1.2、std::array 简要说明

  事实上,库里也有一个array相关链接。其相关使用和数组一致,细微之处在于多了迭代器的各接口。

在这里插入图片描述
  
  那么有一个问题:既然有了数组,为什么还要单独创建一个array的类?

	Array<int>  a1;
	int arr[5];

在这里插入图片描述

  实际上,主要的区别在于:对越界的检查。
  Array<int> a1;:属于函数调用,只要越界,就能检查到。
  int arr[5];:属于指针解引用 ,其越界检查属于设岗抽查,且只针对越界写,越界读不检查。
  
  
  
  

2、模板的特化

  🎯你能解释一下什么是模板特化吗?它在C++模板编程中有什么作用?
  🎯类模板特化的语法与函数模板特化有什么不同?
  🎯你能解释一下全特化(full specialization)和偏特化(partial specialization)的区别吗?在什么情况下你会选择使用全特化,什么情况下选择使用偏特化?
  🎯模板特化与函数重载之间有什么关系和区别?在什么情况下你会选择使用模板特化而不是函数重载?
  🎯如果一个类模板被特化了,那么它的基类模板会如何受到影响?在特化一个类模板时,如何处理与基类模板的关系?
  
  
  

2.1、基本介绍

  1)、问题引入
  如下,我们写一个Less函数模板,用于比较不同类型大小:

template<class T>
bool Less(T left, T right)
{
	return left < right;
}

int main()
{
	// 可以比较,结果正确
	cout <<"Less(1, 2):" << Less(1, 2) << endl;

	// 可以比较,结果正确:date类中比较运算符的实现见下述
	Date d1(2023, 4, 29);
	Date d2(2023, 6, 19);
	cout << "Less(d1, d2):" << Less(d1, d2) << endl;

	// 可以比较,结果错误
	Date* p1 = &d1;
	Date* p2 = &d2;
	cout << "Less(p1, p2):" << Less(p1, p2) << endl;

	return 0;
}

  可以发现:Less适用于绝对多数场景,但是在特殊场景下会得到错误的结果。如上述Less(p1, p2),对此分析,这是因为p1p2为指针类型,指向的是Date对象的地址,我们期望Less函数内部比较的是p1p2指向的对象内容(d1d2),但实际比较的是p1p2指针的地址。在这里插入图片描述
  针对上述情况,就需要用到模板特化即:在原模板类的基础上,针对特殊类型所进行特殊化的实现方式。
  
  
  上述演示例子的,关于date类中比较运算符的实现:(完整版见类和对象三

struct Date
{
	Date(int year, int month, int day)
		:_year(year)
		, _month(month)
		, _day(day)
	{}

	bool operator>(const Date& d) const
	{
		if ((_year > d._year)
			|| (_year == d._year && _month > d._month)
			|| (_year == d._year && _month == d._month && _day > d._day))
		{
			return true;
		}
		else
		{
			return false;
		}
	}

	bool operator<(const Date& d) const
	{
		if ((_year < d._year)
			|| (_year == d._year && _month < d._month)
			|| (_year == d._year && _month == d._month && _day < d._day))
		{
			return true;
		}
		else
		{
			return false;
		}
	}

	int _year;
	int _month;
	int _day;
};

  模板特化中分为函数模板特化类模板特化
  1、函数模板特化: 函数模板特化是指为函数模板提供针对特定类型的替代实现。当函数模板被特化时,对于特化类型,编译器将使用特化版本的函数,而不是通用版本的函数。
  2、类模板特化: 类模板特化是针对类模板的特定类型提供特定的类定义。与函数模板特化类似,类模板特化允许你为某个特定的类型定制类模板的行为。
  
  
  
  
  
  
  

2.2、函数模板特化

  1)、使用说明

template<class T>
bool Less(T left, T right)
{
	return left < right;
}

  仍旧是上述例子,我们Less函数模板进行特化处理:

template<class T>
bool Less(T left, T right)
{
	return left < right;
}

template<>
bool Less<Date*>(Date* left, Date* right)
{
	return *left < *right;
}

  结果如下:
在这里插入图片描述

  函数模板的特化步骤:
  1、必须要先有一个基础的函数模板。
  2、关键字template后面接一对空的尖括号<>
  3、函数名后跟一对尖括号,尖括号中指定需要特化的类型。
  4、函数形参表:必须要和模板函数的基础参数类型完全相同,如果不同编译器可能会报一些奇怪的错误。
  
  当然我们也可直接使用非模板函数: (这里有一个模板参数的匹配原则,相关内容见模板初阶章节)

bool Less(Date* left, Date* right)
{
	return *left < *right;
}

  
  
  
  
  
  
  

2.3、类模板特化

2.3.1、基本说明

  除了函数模板特化,类模板也可以根据需求进行特化处理,以下为相关演示:

namespace myless
{
	//类模板
	template<class T>
	struct less
	{
		bool operator()(const T& val1, const T& val2)
		{
			return val1 < val2;
		}
	};
}

void test06()
{
	Date d1(2023, 4, 29);
	Date d2(2023, 6, 19);
	myless::less<Date> lessFun1;
	cout << lessFun1(d1, d2) << endl;

	Date* p1 = &d1;
	Date* p2 = &d2;
	myless::less<Date*> lessFun2;
	cout << lessFun2(p1, p2) << endl;
}

在这里插入图片描述

  
  同样定义一个类模板,当我们传入参数不同时,存在场景使用错误,因此类模板中也需要特化处理。

namespace myless
{
	//类模板
	template<class T>
	struct less
	{
		bool operator()(const T& val1, const T& val2)
		{
			return val1 < val2;
		}
	};

	template<>//注意模板特化需要处理的地方
	struct less<Date*>//
	{
		bool operator()(Date* val1, Date* val2)//
		{
			return *val1 < *val2;
		}
	};

}

在这里插入图片描述
  
  
  
  
  

2.3.2、用途举例

  我们以优先级队列来举例演示:
  分别用date类构建两个优先级队列,根据之前所学,优先级队列实则是以堆排序数据的,因此我们将相同的date数据传入优先级队列中,再分别拿出打印:

#include<queue>
void test07()
{
	std::priority_queue<Date,vector<Date>,myless::less<Date>> pq1;
	std::priority_queue<Date*, vector<Date*>, myless::less<Date*>> pq2;
	

	pq1.push(Date(2023, 4, 29));pq1.push(Date(2023, 6, 19));pq1.push(Date(2023, 3, 07));pq1.push(Date(2023, 4, 20));
	pq1.push(Date(2023, 9, 18));pq1.push(Date(2023, 7, 11));pq1.push(Date(2023, 8, 24));
	while (!pq1.empty())
	{
		cout << pq1.top();
		pq1.pop();
	}
	
	cout << "________________________________________________" << endl;

	pq2.push(new Date(2023, 4, 29)); pq2.push(new Date(2023, 6, 19)); pq2.push(new Date(2023, 3, 07)); pq2.push(new Date(2023, 4, 20));
	pq2.push(new Date(2023, 9, 18)); pq2.push(new Date(2023, 7, 11)); pq2.push(new Date(2023, 8, 24));
	while (!pq2.empty())
	{
		cout << *(pq2.top());
		pq2.pop();
	}
}

  结果如下:可看到,在没有对模板进行特化处理时,以Date*构建出的优先级队列pq2,其结果非按照大堆排序,实则排序的是new出来的地址空间。
在这里插入图片描述

namespace myless
{
	//类模板
	template<class T>
	struct less
	{
		bool operator()(const T& val1, const T& val2)
		{
			return val1 < val2;
		}
	};

	//template<>
	//struct less<Date*>
	//{
	//	bool operator()(Date* val1, Date* val2)
	//	{
	//		return *val1 < *val2;
	//	}
	//};

}

  
  
  

2.3.3、分类:全特化、偏特化

  1)、全特化、偏特化举例
  以下述类模板举例:

template<class T1,class T2>
class Data
{
public:
	Data()
	{
		cout << "Data<T1,T2>" << endl;
	}
private:
	T1 _d1;
	T2 _d2;
};

  全特化: 将模板参数列表中所有的参数都确定化。

//全特化
template<>
class Data<int, char>
{
public:
	Data()
	{
		cout << "Data<int,char>" << endl;
	}
};

  偏特化: 任何针对模版参数进一步进行条件限制设计的特化版本。
  偏特化演示一:部分特化,将模板参数类表中的一部分参数特化。

template<class T1>//注意偏特化中,这里模板参数需要给出
class Data<T1, int>
{
public:
	Data() { cout << "Data<T1, int>" << endl; }
};

  偏特化演示二:偏特化不仅仅是指特化部分参数,而是针对模板参数更进一步的条件限制所设计出来的一个特化版本。

template<class T1,class T2>
class Data<T1*, T2*>//这也是偏特化的一种,这里限制了参数必须是指针类型
{
public:
	Data() { cout << "Data<T1*, T2*>" << endl; }
};

  相关结果:
在这里插入图片描述

  

template<class T1, class T2>
class Data<T1&, T2&>//模板参数还可以是引用
{
public:
	Data() { cout << "Data<T1*, T2*>" << endl; }
};

template<class T1, class T2>
class Data<T1*, T2&>//也可以是二者结合
{
public:
	Data() { cout << "Data<T1*, T2&>" << endl; }
};

template<class T1, class T2>
class Data<T1, T2&>//也可以是二者结合
{
public:
	Data() { cout << "Data<T1, T2&>" << endl; }
};

在这里插入图片描述
  
  
  
  
  
  
  
  

3、模板的分离编译

3.1、问题说明

3.1.1、例子

  我们以vector来举例说明:

  1、在vector.h文件中,类的声明: 此处以insertpush_back来举例说明,我们将其在类中声明,在类外定义。

//vector.h文件
namespace myvector
{
	template<class T>
	class vector
	{
	public:
		//这里为了观察省去一部分成员函数
		//……
		
		//尾删数据
		void pop_back()
		{
			assert(_finish > _start);
			_finish--;
		}

		iterator insert(iterator pos, const T& val);
		void push_back(const T& val);


	private:
		iterator _start;
		iterator _finish;
		iterator _end_of_storage;
	};
}

  
  
  2、在vector.cpp文件中,类的定义:

  注意这里的各种写法:
  1、vector<T>::push_backvector<T>::insert :类外使用要注意指定类域。如果不展开命名空间,则为myvector::vector<T>::push_back
  2、typename vector<T>::iterator,加上该关键字是为了区别后面iterator是类中的一个类型,而非静态类成员,因为静态类成员也可以使用类域直接访问。

  见模板初阶关于typename关键字的介绍: 当模板代码涉及到嵌套依赖类型时(例如,模板类的成员模板),typename用于告诉编译器接下来的名称是一个类型名,而不是一个成员变量或函数。

//vector.cpp文件
namespace myvector
{
	template<class T>
	typename vector<T>::iterator typename vector<T>::insert(typename vector<T>::iterator pos, const T& val)
	//vector<T>::iterator、vector<T>::insert 类外使用,要注意指定类域
	{
		assert(pos >= _start && pos <= _finish);

		if (_finish == _end_of_storage)
		{
			size_t len = pos - _start;
			reserve(capacity() == 0 ? 4 : capacity() * 2);
			pos = _start + len;
		}

		iterator end = _finish - 1;
		while (end >= pos)
		{
			*(end + 1) = *end;
			--end;
		}
		*pos = val;
		++_finish;
		return pos;
	}

	template<class T>
	void vector<T>::push_back(const T& val)
	{
		//检查
		if (_finish == _end_of_storage)
		{
			reserve(capacity() == 0 ? 4 : capacity() * 2);
		}
		//插入
		*_finish = val;
		++_finish;
	}
}


  
  

3.1.2、测试结果及原因解释

  3、在test.cpp文件中,以下为测试代码:

//test.cpp
#include"vector.h"
void test09()
{
	myvector::vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	for (size_t i = 0; i < v1.size(); ++i)
	{
		cout << v1[i] << " ";
	}
}

  我们运行上述代码:可发现结果报错,且是链接错误。但当我们没使用insertpush_back声明和定义分离的这两个函数时,成功运行。
在这里插入图片描述

  原因解释:
  1、test.cpp文件中包含了头文件vector.h,头文件在编译阶段会展开,而故头文件中定义的函数operator[]、size等,后续vector<int> v1实例化时,这些成员函数都跟随实例化,也就有了具体定义,那么编译阶段能够直接确定地址。

  2、insert、push_back声明和定义分离vector.h中只有二者声明,而test.cpp中我们使用这两个函数,那么即使头文件被展开,在编译阶段我们没有得到二者的确切地址,故只能在链接阶段所有.obj文件汇总后去寻找相关地址。

  3、但我们得到的结果是报错,说明链接阶段没有找到insert、push_back的地址,这是因为声明定义分离后,其中模板参数T无法确定,即二者没有实例化,相应地址也就没有进入符号表,故链接出错。
  
  
  
  
  
  
  

3.2、解决方案:显式实例化

  关于模板声明定义分离解决方案:
  1、将声明和定义放到同一个文件里,比如 “xxx.hpp” 或者xxx.h,在该基础上,类里声明、类外实现函数具体方法。

  2、模板定义的位置显式实例化,如下:这种写法存在的一个缺陷是把类型写死了

//vector.cpp文件
namespace myvector
{
	template<class T>
	typename vector<T>::iterator typename vector<T>::insert(typename vector<T>::iterator pos, const T& val)//vector<T>::iterator、vector<T>::insert 类外使用,要注意指定类域
	{
		assert(pos >= _start && pos <= _finish);

		if (_finish == _end_of_storage)
		{
			size_t len = pos - _start;
			reserve(capacity() == 0 ? 4 : capacity() * 2);
			pos = _start + len;
		}

		iterator end = _finish - 1;
		while (end >= pos)
		{
			*(end + 1) = *end;
			--end;
		}
		*pos = val;
		++_finish;
		return pos;
	}

	template<class T>
	void vector<T>::push_back(const T& val)
	{
		//检查
		if (_finish == _end_of_storage)
		{
			reserve(capacity() == 0 ? 4 : capacity() * 2);
		}
		//插入
		*_finish = val;
		++_finish;
	}

	//针对整个类进行显示实例化的的方法:
	template
	vector<int>;

	template
	vector<double>;
}

  
  
  模板小结:
  优点:
  1、模板复用了代码,节省资源,更快的迭代开发,C++的标准模板库(STL)因此而产生
  2、增强了代码的灵活性

  缺陷:
  1、模板会导致代码膨胀问题,也会导致编译时间变长
  2、出现模板编译错误时,错误信息非常凌乱,不易定位错误
  
  
  
  
  
  
  
  
  
  
  

Fin、共勉。

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值