模板初阶(c++)

目录

1.泛型编程

2.函数模板

2.1定义:

2.2函数模板的格式

2.3函数模板的的原理

2.4函数模板的实例化

2.5模板参数的匹配原则

3.类模板

3.1类模板的定义格式

3.2类模板的实例化

1.泛型编程

告诉编译器一个模子,让编译器根据不同的的类型利用模子来生成代码。

2.函数模板

2.1定义:

函数模板是代表了一个函数家族,该函数模板与类型无关,在使用的时被参数化,根据实参类型产生对应类型的函数。(注意这是一个新的函数而不是原来的函数模板,函数模板不是函数,函数模板参数化之后才是函数)

2.2函数模板的格式

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

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

注意typename是用来定义模板参数的关键字,也可以使用class。

2.3函数模板的的原理

函数模板是一个模子,它本身并不是函数,是编译器用使用方式产生特定具体类型的函数的模子,所以模板就是将本来我们应该做的事交给了编译器。在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演出对应的函数以供调用。

2.4函数模板的实例化

使用不同的参数使用函数模板时称为函数模板的的实例化。就模板参数实例化就分为:隐式实例化和显示实例化。

1.隐式实例化:让编译器根据实参推演模板参数的实例化。

2.显示实例化:在函数名后面的**<>中指定模板参数**的实际类型

cpp 复制代码
#define  _CRT_SECURE_NO_WARNINGS
#include <iostream>
using namespace std;
//下面是模板函数
template <class t>
t add(t a, t b)
{
	return a + b;
}
int main()
{
	int a1 = 1;
	double a2 = 1.1;
	cout << add(a1, (int)a2) << endl;//这个是隐式实例化,在传参的时候自动推导出t的类型
	cout << add<double>(a1, a2) << endl;//这个是显示实例化,直接就指定了t类型为double,不一样的就强转。
	//对于形参没有t的模板,只能用显示实例化,因为传参的时候并不能确定t的类型,那么返回值的类型就无法确定

	return 0;
}

2.5模板参数的匹配原则

1.一个非模板函数可以和一个同名的函数模板同时存在,而且改函数模板还可以被实例化。

2.如果一个函数模板跟一个或多个非模板的同名函数同时存在,如果其他条件都一样的话,在调用的时候会遵循先匹配,如果适配度也一样就调用非模板的函数,而不是用函数模板来产生一个实例。

cpp 复制代码
#define  _CRT_SECURE_NO_WARNINGS
#include <iostream>
using namespace std;
int add(int a, int b)
{
	return a + b;
}
template <class t>
t add(t a, t b)
{
	return a + b;
}

int main()
{
	int a1 = 1;
	double a2 = 1.1;
	cout << add(a1, (int)a2) << endl;//这个是隐式实例化,在传参的时候自动推导出t的类型
	cout << add<double>(a1, a2) << endl;//这个是显示实例化,直接就指定了t类型为double,不一样的就强转。
	//对于形参没有t的模板,只能用显示实例化,因为传参的时候并不能确定t的类型,那么返回值的类型就无法确定
	//模板参数的匹配原则:(调用最匹配的,相同匹配的调用简单的)
	//对于非模板和同名函数的模板,如果其他的条件都相同,在调用时会优先调用非模板,如果模板可以产生一个具有更寒匹配的函数,那就调用模板
	cout << add(a1, a1) << endl;//这里调用的是现成的函数而不是模板
	return 0;
}

3.注意:模板函数不能自动类型转换,但是普通函数可以。

3.类模板

3.1类模板的定义格式

template <class T1,class T2.......>

class 类模板名

{

类内成员的定

义;

};

cpp 复制代码
#define  _CRT_SECURE_NO_WARNINGS
#include <iostream>
using namespace std;
//下面是类模板
template <class t>
class Stack
{
public:
	t* _array;
	size_t _size;
	size_t _capacity;
	Stack(size_t cap=4)
	{
		_array = new t[cap];//new一个t类型的数组
		_capacity = cap;
		_size = 0;
	}
	void pushstack(const t& date);//声明
};
template <class t>
void Stack<t>::pushstack(const t& date)//定义
{
	_array[_size++] = date;
}
int main()
{
	
	Stack <int> st1;//创建的时候必须添加模板参数<int>
	cout << st1._array << endl;
	Stack <double> st2;//模板参数<double>
	cout << st2._array << endl;
	return 0;
}

注意:类模板中函数放到类外定义的时候需要加上模板参数列表。

3.2类模板的实例化

类模板的实例化跟函数模板的实例化不同,类模板实例化需要在类模板的的名字后面加上<>,然后将实例化的类型放到<>中,同样类模板名字把并不是真正的类,实例化之后才是一个类名,然后用这个类来创建对象。可以理解为:模板名<实例化类型列表> 这个整体是一个类。

相关推荐
不会Hello World的小苗3 分钟前
Java——链表(LinkedList)
java·开发语言·链表
lsx20240615 分钟前
Perl 面向对象编程指南
开发语言
Allen Bright36 分钟前
【Java基础-46.3】Java泛型通配符详解:解锁类型安全的灵活编程
java·开发语言
JANGHIGH42 分钟前
c++ std::list使用笔记
c++·笔记·list
画个逗号给明天"1 小时前
C++STL容器之list
开发语言·c++
hrrrrb1 小时前
【Java】Java 常用核心类篇 —— 时间-日期API(上)
java·开发语言
小突突突1 小时前
模拟实现Java中的计时器
java·开发语言·后端·java-ee
七禾页话1 小时前
垃圾回收知识点
java·开发语言·jvm
web137656076431 小时前
Scala的宝藏库:探索常用的第三方库及其应用
开发语言·后端·scala
煤炭里de黑猫2 小时前
Lua C API :使用 lua_tonumber 函数从 Lua 栈中提取数值
开发语言·lua