一、什么是泛型编程
当我们设计函数或者类时,有时候需要对应不同数据类型编写相同的代码,这样的话不仅有代码冗余,而且更加的加大程序员开发事件,降低开发效率,因此泛型编程就是解决此类情况----不同的数据类型可以重用一个函数,或者可以创建不同数据类型的相同类。
二、泛型的声明
template <typename name ...>
样例:
#include <iostream>
using namespace std;
template <typename _Tp>
_Tp Max(_Tp a,_Tp b)
{
return a > b ? a : b;
}
int main()
{
cout << Max(1, 2) << endl; //传入两个整型
cout << Max(10.55, 2.33) << endl;//传入两个浮点型
return 0;
}
三、泛型的特性
1、显示和隐式传值
2、传常量
注意:当模板规定变量类型时调用的时候必须显示调用,并且传参只能传常量。
3、默认情况
4、普通函数和模板函数的优先级
可以看到普通函数的优先级是高于模板函数的
四、类模板
1、模板类样例
#include <iostream>
using namespace std;
template <typename _Ty1,typename _Ty2>
class Data
{
public :
Data();
Data(_Ty1 one, _Ty2 two):m_one(one),m_two(two){}
~Data();
void printData();
private:
_Ty1 m_one;
_Ty2 m_two;
};
//用到类型的地方必须使用 类名<类型> 的方式使用
template <typename _Ty1,typename _Ty2>
void Data<typename _Ty1, typename _Ty2>::printData()
{
cout << m_one << " " << m_two << endl;
}
int main()
{
Data<int, string>* data = new Data<int, string>(1,"abc");//必须显示创建类对象
data->printData();
return 0;
}
template<typename _Ty1, typename _Ty2>
Data<_Ty1, _Ty2>::Data()
{
}
template<typename _Ty1, typename _Ty2>
Data<_Ty1, _Ty2>::~Data()
{
}
注意:模板类的定义和实现必须在一个文件中,**不能.h中实现类的定义在.cpp文件中实现类方法等。**模板类的声明必须显示创建
2、类模板的特化
当我们需要处理特殊数据时就需要类模板的特化(因为有些设备不一样,对于相同的数据要进行不同方法进行处理)
#include <iostream>
using namespace std;
template <typename _Ty1,typename _Ty2>
class Data
{
public :
Data();
Data(_Ty1 one, _Ty2 two):m_one(one),m_two(two)
{
cout << "这是原本模板类" << endl;
}
~Data();
private:
_Ty1 m_one;
_Ty2 m_two;
};
//完全特例
template <>
class Data<int,int>
{
public:
Data(int one, int two) :m_one(one), m_two(two)
{
cout << "这是特例类" << endl;
}
private:
int m_one;
int m_two;
};
int main()
{
Data<int, int>* data1 = new Data<int, int>(1,1);
Data<int, string>* data2 = new Data<int, string>(1,"abc");
return 0;
}
五、可变模板
1、可变模板写法
template<typename ...Args>
void func(Args...args)
{
.....
}
2、参数包的展开
- 递归展开
#include <iostream>
using namespace std;
//递归终止函数
void func()
{
cout << endl;
}
//展开函数
template <typename T,typename ...Args>
void func(T value,Args...args)
{
cout<<value<< " ";//查看个数
cout << sizeof...(args) << endl;
func(args...);//最后参数包里面的个数为0时就匹配我们外边的func终止函数来结束递归展开
}
int main()
{
func(1, 2, 3);
return 0;
}
- 逗号表达式展开
#include <iostream>
using namespace std;
//展开函数
template <typename ...Args>
void func(Args...args)
{
int arr[] = { args... };
for(auto c:arr)
{
cout << c << " ";
}
cout << endl;
}
int main()
{
func(1,2,3,4);
return 0;
}