C++——模板(函数模板、类模板)

函数模板

函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定类型版本

模板格式:

template<typename T1, typename T2,......,typename Tn> 

返回值类型 函数名(参数列表){}

template<typename T> 
void Swap( T& left,  T& right) {    
    T temp = left;    
    left = right;    
    right = temp; 
}

注::typename是用来定义模板参数关键字,也可以使用class(切记:不能使用struct代替class)。

模板原理:编译器需要根据传入的实参类型来推演生成对应类型的函数

函数模板的实例化: 

用不同类型的参数使用函数模板。

分为隐式实例化显式实例化。


隐式实例化:让编译器根据实参推演模板参数的实际类型

template<class T> 
T Add(const T& left, const T& right) {   //用引用传参的效率是提高效率
     return left + right; 
}
 
int main() {    
    int a1 = 10, a2 = 20;    
    double d1 = 10.0, d2 = 20.0;    
    Add(a1, a2);    
    Add(d1, d2);        
/*     注意:在模板中,编译器一般不会进行类型转换操作,因为一旦转化出问题,编译器就需要背黑锅     Add(a1, d1);    */        
// 此时有两种处理方式:1. 用户自己来强制转化 2. 使用显式实例化    

    Add(a1, (int)d1);    //自己强制转换
    return 0;
}

显式实例化:函数名后面加<>注明函数类型

template<class T> 
T Add(const T& left, const T& right) {   
     return left + right; 
}
 
int main() {    
    int a1 = 10, a2 = 20;    
    double d1 = 10.0, d2 = 20.0;    
    Add<int>(a1, d1); //显式实例化
   //如果类型不匹配,编译器会尝试进行隐式类型转换,如果无法转换成功编译器将会报错
    Add<>(a1, d1); //这样也可以但是是隐式转换,不属于显式转化
    return 0;
}

注:函数模板不属于函数;因为类似飞机模型能是飞机吗?


对于正常函数,是会进行自己的类型隐式转化的。

// 专门处理int的加法函数 
int Add(int left, int right) {   
     return left + right; 
}
 

 
void Test() {    
    Add(1, 2.0);       //进行类型隐式转化,能转化转化;转换不了,就报错。
}

还有一种解决方式:利用函数模板含有这两种参数

//显式实例化:函数名之后加<类型>
template<class T1,typename T2> 
T1 Add(const T1& left, const T2& right) {
	return left + right;
}

int main() {
	int a1 = 10, a2 = 20;
	double d1 = 10.0, d2 = 20.0;
	Add(a1, d1); //	
        return 0;
}

模板存在最大原因:懒!!!!

模板参数的匹配原则:

  • 一个非模板函数可以和一个同名的函数模板同时存在,而且该函数模板还可以被实例化为这个非模板函数。
  • 对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例。如果模板可以产生一个具有更好匹配的函数, 那么将选择模板。
  • 模板函数不允许自动类型转换,但普通函数可以进行自动类型转换。
// 专门处理int的加法函数 
int Add(int left, int right) {   
     return left + right; 
}
 
// 通用加法函数 
template<class T> 
T Add(T left, T right) {    
    return left + right; 
}
 
void Test() {    
    Add(1, 2);       // 与非模板函数匹配,编译器不需要特化    
    Add<int>(1, 2);  // 调用编译器特化的Add版本 
}

注:没有重定义,也没有函数重载;因为编译器生成的名字:add<int>     函数名<类型>


类模板

//类模板
//C语言方式
typedef int DataType;
class  SeqList{
public:
	//我们之前的方式
	SeqList(size_t capacity = 10) {
		_array = new DataType[capacity];
		_capacity = capacity;
		_size = 0;
	}
	
	//初始化列表
	SeqList(size_t capacity = 10) 
		:_array(new DataType[capacity])
		, _capacity(capacity)
		,_size (0)
	{//赋初值(可以不要)
		_array = new DataType[capacity];
		_capacity = capacity;
		_size = 0;
	}

	~SeqList() {
		if (_array) {
			delete[] _array;
			_array= nullptr;
			_capacity = 0;
			_size = 0;
		}
	}
private:    
	DataType* _array;    
	size_t _size;    
	size_t _capacity;
};

因为上一个只能放 int 类型的数据,要是别的属性,就要,很麻烦,于是进行类模板

//类模板
template<class T>
class  SeqList{
public:
	//我们之前的方式
	SeqList(size_t capacity = 10) {
		_array = new DataType[capacity];
		_capacity = capacity;
		_size = 0;
	}
	
	//初始化列表
	SeqList(size_t capacity = 10) 
		:_array(new DataType[capacity])
		, _capacity(capacity)
		,_size (0)
	{}

	~SeqList() {
		if (_array) {
			delete[] _array;
			_array= nullptr;
			_capacity = 0;
			_size = 0;
		}
	}
private:    
	T* _array;    
	size_t _size;    
	size_t _capacity;
};

这是在其中加入函数:

//类模板
template<class T>
class  SeqList{
public:
	//我们之前的方式
	SeqList(size_t capacity = 10) {
		_array = new DataType[capacity];
		_capacity = capacity;
		_size = 0;
	}
	
	//初始化列表
	SeqList(size_t capacity = 10) 
		:_array(new DataType[capacity])
		, _capacity(capacity)
		,_size (0)
	{}

	~SeqList() {
		if (_array) {
			delete[] _array;
			_array= nullptr;
			_capacity = 0;
			_size = 0;
		}
	}

        void PushBack(const T& data) //加入的函数


private:    
	T* _array;    
	size_t _size;    
	size_t _capacity;
};

//注意:类模板中函数放在类外进行定义时,需要加模板参数列表 
//还有函数名字  SeqList<T>::PushBack   很关键
template<class T>
void SeqList<T>::PushBack(const T& data) {  //加入的函数
    _array[_size++] = data;
}

注意:类模板中函数放在类外进行定义时,需要加模板参数列表。函数名很关键,因为类模板也不是类,实例化之后才是类。


类模板的实例化:

类模板实例化:类模板<类型>(类模板实例化需要在类模板名字后跟<>,然后将实例化的类型放在<> 中)类模板名字不是真正的,而实例化的结果真正的

 // SeqList类名,SeqList<int>才是类型 
SeqList<int> s1;

这时候 s1 可以调用类中的函数了。

s1.PushBack; //刚加入的函数

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值