C++模板template原理解析

目录
  • 前言
  • 1. 函数模板
    • 1.1函数模板的概念
    • 1.2函数模板的格式
    • 1.3 函数模板的原理
    • 1.4 函数模板的实例化
      • 1.4.1 隐式实例化
      • 1.4.2 显式实例化
    • 1.5 模板参数的匹配原则
  • 2. 类模板
    • 2.1 类模板的定义格式
    • 2.2 类模板的实例化

前言

在学习模板之前我们首先要了解泛型编程。泛型编程是一种编程风格,其中算法以尽可能抽象的方式编写,而不依赖于将在其上执行这些算法的数据形式。泛型编程只编写与类型无关的通用代码,是代码复用的一种手段。本节学习的模板是泛型编程的基础。

模板分为:函数模板和类模板

1. 函数模板

1.1函数模板的概念

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

1.2函数模板的格式

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

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

//函数模板
void Swap(T& left, T& right)
{
T tmp = left;
left = right;
right = tmp;
}

其中typename是用来定义模板参数的关键字,也可以使用class.(但是不能使用struct代替class).

1.3 函数模板的原理

函数模板是一个蓝图,其本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具,所以其实模板就是将本来应该我们做的重复的事情交给了编译器。

我们以Swap()交换函数来进行举例。如何实现一个通用的交换函数呢?

void Swap(int& left, int& right) {
int temp = left;
left = right;
right = temp;
}
void Swap(double& left, double& right) {
double temp = left;
left = right;
right = temp;
}
int main()
{
int a = 0;
int b = 1;
double c = 2.2;
double d = 3.3;
Swap(a, b);
Swap(c, d);
return 0;
}

在这段代码中,我们使用到了函数重载,但是仍然有几个不好的地方:

  • 1、重载的函数仅仅是类型不同,代码复用率比较低,只要有新的类型出现时,就需要我们自己新增对应的函数。
  • 2、代码的可维护行比较低,一个出错可能所有的重载均出错。

因此,介于上面可能发生的问题,C++便使用函数模板来解决这个问题。

根据上面的模板结构,Swap()函数用模板的方法来写如下所示:

//Swap()函数
//template<typename T>
template<class T>
void Swap(T& left, T& right)
{
T tmp = left;
left = right;
right = tmp;
}

我们使用模板解决了以上两个问题。其中,编译器对特定具体类型的函数会调用相对应类型的Swap函数。

在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。

比如:当用int类型使用函数模板时,编译器通过对实参类型的推演,将T确定为int类型,然后产生一份专门处理int类型的代码,对于其他类型也是如此

1.4 函数模板的实例化

用不同类型的参数使用函数模板时,成为函数模板的实例化。模板参数实例化分为:隐式实例化和显式实例化。

1.4.1 隐式实例化

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

template<class T>
T Add(const T& left, const T& right)
{
return left + right;
}
int main()
{
int a1 = 10, a2 = 20;
double d1 = 20.0, d2 = 10.0;
Add(a1, a2);
Add(d1, d2);
return 0;
}

其中Add(a1,a2)和Add(d1,d2)就是隐式实例化。编译器会根据实参推演模板参数的实际类型。

template<class T>
T Add(const T& left, const T& right)
{
return left + right;
}
int main()
{
int a1 = 10;
double d1 = 20.0;
Add(a1, d1);
return 0;
}

注意:上述代码是不能通过编译的,因为在编译期间,当编译器看到该实例化时,需要推演其实参类型来确定模板参数的具体类型,但是通过实参a1将T推演为int,通过实参d1将T推演为double,由于模板参数列表中只有一个T,因此编译器无法确定到底该将T确定为int或者是double类型,从而会报错。(在模板中,编译器一般不会进行类型转换的操作)

此时可以用两种处理方式:

  • 1、用户自己来强制转换
  • 2、使用显式实例化
int main()
{
int a1 = 10;
double d1 = 20.0;
Add(a1, (int)d1);//用户自己来强制转换
return 0;
}

1.4.2 显式实例化

显式实例化:在函数名后的<>中指定模板参数的实际类型。

template<class T>
T Add(const T& left, const T& right)
{
return left + right;
}
int main()
{
int a1 = 10;
double d1 = 20.0;
Add<int>(a1, d1);//显示实例化成int
Add<double>(a1, d1);//显示实例化成double
return 0;
}

如果类型不匹配,编译器会尝试进行隐式类型转换,如果无法转换成功编译器将会报错。

1.5 模板参数的匹配原则

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

// 专门处理int的加法函数
int Add(int left, int right) {
return left + right;
}
// 通用加法函数
template<class T> T Add(T left, T right) {
return left + right;
}
int main()
{
Add(1, 2); // 与非模板函数匹配,编译器不需要特化
Add<int>(1, 2); // 调用编译器特化的Add版本
return 0;
}

2.  对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例,如果模板可以产生一个具有更好匹配的函数,那么将选择模板

// 专门处理int的加法函数
int Add(int left, int right) {
return left + right;
}
// 通用加法函数
template<class T1, class T2>
T1 Add(T1 left, T2 right) {
return left + right;
}
int main()
{
//与非函数模板类型完全匹配,不需要函数模板实例化
Add(1, 2);
//模板函数可以生成更加匹配的版本
//编译器根据实参生成更加匹配的Add函数
Add(1, 2.0);
return 0;
}

3. 模板函数不允许自动类型转换,但普通函数可以进行自动类型转换

2. 类模板

2.1 类模板的定义格式

template<class T1, class T2, ..., class Tn>
class 类模板名
{
// 类内成员定义
};
// 动态顺序表
// 注意:Vector不是具体的类,是编译器根据被实例化的类型生成具体类的模具
template<class T>
class Vector
{
public:
Vector(size_t capacity = 10)
: _pData(new T[capacity])
, _size(0)
, _capacity(capacity)
{}
// 使用析构函数演示:在类中声明,在类外定义。
~Vector();
void PushBack(const T& data);
void PopBack();
// ...
size_t Size() { return _size; }
T& operator[](size_t pos)
{
assert(pos < _size);
return _pData[pos];
}
private:
T* _pData;
size_t _size;
size_t _capacity;
};
// 注意:类模板中函数放在类外进行定义时,需要加模板参数列表
template <class T>
Vector<T>::~Vector()
{
if (_pData)
delete[] _pData;
_size = _capacity = 0;
}

2.2 类模板的实例化

类模板实例化与函数实例化不同,类模板实例化需要在类模板名字后跟<>,然后将实例化的类型放在<>中即可,类模板名字不是真正的类,而实例化的结果才是真正的类。

// Vector类名,Vector<int>才是类型
Vector<int> s1;
Vector<double> s2;

到此这篇关于 C++模板template原理的文章就介绍到这了,更多相关 C++模板template内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++模板template用法小结(推荐)

    引言 模板(Template)指C++程序设计设计语言中采用类型作为参数的程序设计,支持通用程序设计.C++ 的标准库提供许多有用的函数大多结合了模板的观念,如STL以及IO Stream. 函数模板 在c++入门中,很多人会接触swap(int&, int&)这样的函数类似代码如下: void swap(int&a , int& b) { int temp = a; a = b; b = temp; } 但是如果是要支持long,string,自定义class的swap函

  • C++设计模式之模板方法模式(TemplateMethod)

    模板方法模式使用继承来实现模式的功能,在基类使用一个方法来定义算法的各个步骤,这些步骤(方法)的具体实现会放到子类中,通过这样来实现不同算法对象的算法拼合,完成该对象整体算法的实现. 作用 模板方法中定义了具体操作中所使用算法的各个步骤,并将其实现交由子类完成,从而实现多种不同的功能: 类视图 实现 class Lunch { public: Lunch(){} virtual ~Lunch(){} void feed() { cooking(); eating(); cleaning(); }

  •  C++模板template原理解析

    目录 前言 1. 函数模板 1.1函数模板的概念 1.2函数模板的格式 1.3 函数模板的原理 1.4 函数模板的实例化 1.4.1 隐式实例化 1.4.2 显式实例化 1.5 模板参数的匹配原则 2. 类模板 2.1 类模板的定义格式 2.2 类模板的实例化 前言 在学习模板之前我们首先要了解泛型编程.泛型编程是一种编程风格,其中算法以尽可能抽象的方式编写,而不依赖于将在其上执行这些算法的数据形式.泛型编程只编写与类型无关的通用代码,是代码复用的一种手段.本节学习的模板是泛型编程的基础. 模板

  • C++设计模式编程中Template Method模板方法模式的运用

    准备一个抽象类,将部分逻辑以具体方法以及具体构造子的形式实现,然后声明一些抽象方法来迫使子类实现剩余的逻辑.不同的子类可以以不同的方式实现这些抽象方法,从而对剩余的逻辑有不同的实现.这就是模版方法模式的用意. 很多人可能没有想到,模版方法模式实际上是所有模式中最为常见的几个模式之一,而且很多人可能使用过模版方法模式而没有意识到自己已经使用了这个模式.模版方法模式是基于继承的代码复用的基本技术,模版方法模式的结构和用法也是面向对象设计的核心. 模版方法模式需要开发抽象类和具体子类的设计师之间的协作

  • C++中的模板template小结

    函数模板 我们可以把函数模板当做一种特殊的函数,里面的参数类型可以是任意类型,这样的话我们就可以减少重复定义,从而让这个函数模板自动适应不同的参数类型,也就是说函数可以适应多种类型的参数,例如double.int或者类什么的. C++为了实现上面的功能,引入了template这个概念.我们可以把template当成是一种特殊的类型参数,并且也可以在函数里当做参数传递,心里面把它当做int什么的就行了. 使用类型参数声明函数模板的格式如下所示: template <class identifier

  • C++ Template 基础篇(一):函数模板详解

    Template所代表的泛型编程是C++语言中的重要的组成部分,我将通过几篇blog对这半年以来的学习做一个系统的总结,本文是基础篇的第一部分. 为什么要有泛型编程 C++是一门强类型语言,所以无法做到像动态语言(python javascript)那样子,编写一段通用的逻辑,可以把任意类型的变量传进去处理.泛型编程弥补了这个缺点,通过把通用逻辑设计为模板,摆脱了类型的限制,提供了继承机制以外的另一种抽象机制,极大地提升了代码的可重用性. 注意:模板定义本身不参与编译,而是编译器根据模板的用户使

  • C++详解非类型模板参数Nontype与Template及Parameters的使用

    目录 非类型类模板参数 非类型函数模板参数 非类型模板参数的限制 非类型模板参数 auto 非类型类模板参数 前一章使用的例子 Stack 使用的是标准库中的容器管理元素,也可以使用固定大小的 std::array,它的优势是内存管理开销更小,数组的大小可以交给用户指定. #include <array> #include <cassert> template<typename T, std::size_t Maxsize> class Stack { private:

  • C++模板Template详解及其作用介绍

    目录 1. 模板 2. 函数模板 2.1 函数模板概念 2.2 函数模板格式 2.3 函数模板原理 2.4 函数模板的实例化 2.5 模板参数的匹配原则 2.6声明定义分离 3. 类模板 3.1 类模板格式 3.2 类模板的实例化 3.3 类模板中函数放在类外进行定义时 4. 模板分离编译 4.1 什么是分离编译 4.2 模板的分离编译 5. 缺省值与返回值 6. 总结 1. 模板 首先模板分为函数模板和类模板 想到模板,就会联想到泛型编程 泛型编程:编写与类型无关的通用代码,是代码复用的一种手

  • C++的template模板中class与typename关键字的区别分析

    在C++模板中,可以使用class或者typename来声明模板参数,那么这两个关键字有什么区别呢? 模板参数声明 对于模板参数声明,这两个参数没有区别,含义是一样的. template class Simple; template class Simple; 上面两行都是声明一个模板类Simple. 表明类型 假如我们有这样一段代码: template void add(const T &acontainer, T &sum) { T::const_iterator iter = con

  • C++中模板(Template)详解及其作用介绍

    目录 概述 函数模板 类模板 模板类外定义成员函数 类库模板 抽象和实例 概述 模板可以帮助我们提高代码的可用性, 可以帮助我们减少开发的代码量和工作量. 函数模板 函数模板 (Function Template) 是一个对函数功能框架的描述. 在具体执行时, 我们可以根据传递的实际参数决定其功能. 例如: int max(int a, int b, int c){ a = a > b ? a:b; a = a > c ? a:c; return a; } long max(long a, l

随机推荐