【C++】模板初阶

简介: 【C++】模板初阶

1. 泛型编程


首先,我们通过一个问题来引入泛型编程的概念。

如果我们想要实现对任何类型都能够适用的swap函数,结合C++的语法,目前能想到的方法就只有利用函数重载的方式,把每一种类型的swap函数都写一遍。代码如下。

void swap(int& a, int& b)
{
  int tmp = a;
  a = b;
  b = tmp;
}
void swap(char& a, char& b)
{
  char tmp = a;
  a = b;
  b = tmp;
}
void swap(double& a, double& b)
{
  double tmp = a;
  a = b;
  b = tmp;
}
//......


显然,我们可以发现很多问题:

  1. 上述重载出来的几个swap函数只有参数类型不同,其余完全相同,造成了代码冗余
  2. 这些预设的swap函数只能支持内置类型,如果出现了一个自定义类型需要使用swap函数的时候,就需要我们重新写一个针对于此自定义类型swap函数。
  3. 代码的可维护行比较低,一个出错可能所有的重载均出错


那能否告诉编译器一个模子,让编译器根据不同的类型利用该模子来生成代码呢?

答案是肯定的。在C++中,是存在这样一个模具,通过给这个模具中填充不同类型,来生成具体类型的代码。这个模具叫做==模板==。

泛型编程:编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础


模板分为函数模板类模板

e0c192587978dc239c427c22a242a50b.png


2. 函数模板


1. 函数模板的概念与格式

概念:函数模板所代表的是执行某一种功能的一类的函数,但是他本身不是函数,只是一个蓝图。在使用时被编译器参数化,根据实参类型产生特定的函数版本。


使用格式:

//格式1
template<typename T1, typename T2, ......>
返回值类型 函数名(参数列表) {}
//格式2
template<class T1, class T2, ......>
返回值类型 函数名(参数列表) {}    

其中,template是定义模板的关键字,typename和class是定义模板参数的关键字

例如,本文开头提出的问题,使用模板改写之后的结果:

template<class T>
void Swap(T& a, T& b)
{
  T tmp = a;
  a = b;
  b = tmp;
}
void Test_Template()
{
  int a = 10, b = 20;
  char c = 'a', d = 'b';
  double e = 1.1, f = 2.2;
  Swap(a, b);
  Swap(c, d);
  Swap(e, f);
}

c6a3163f496594ef4bc35e8099e84528.png

可以看到,使用了一个模板,就可以做到同时支持不同类型参数的功能。


2.函数模板的底层原理

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

编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。比如:当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型也是如此

ce606ec9cd226bfe923b64dbf81b3dc6.png

通过查看汇编代码,可以看到,最终Swap模板实例化出了Swap<int>,Swap<char>,swap<double>三个函数


3.函数模板的实例化

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

1. 隐式实例化:让编译器根据传入的实参自动推演参数的实际类型

template<class T>
T Add(const T a, const T b)
{
  return a + b;
}
void Test1()
{
  int a = 10, b = 20;
  double e = 1.1, f = 2.2;
  cout << Add(a, b) << endl;//会实例化出Add<int>
  cout << Add(e, f) << endl;//会实例化出Add<double>
  cout << Add(a, f) << endl;//此语句不能通过编译
}


上述代码中,前两此调用Add能够被编译器推导出来参数类型,但是第三个,在编译期间,通过实参a将T推演成int类型,通过f将T推演成double类型,但是模板参数列表中只有一个T,因此编译器无法确定此处T将被推演成int还是double而报错。


注意:在模板中,编译器一般不会进行类型转换操作,因为一旦转化出问题,编译器就需要背黑锅


此时,有两种处理方式:

  1. 用户自己把其中一个实参进行强制转换Add((double)a, f)或者Add(a, (int)f)
  2. 使用通过显式实例化


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

template<class T>
T Add(const T a, const T b)
{
  return a + b;
}
void Test1()
{
  int a = 10, b = 20;
  double e = 1.1, f = 2.2;
  cout << Add<int>(a, f) << endl;//实例化成Add<int>
    cout << Add<double>(a, f) << endl;//实例化成Add<double>
}


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


4.模板参数的匹配原则


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

int Add(int a, int b)
{
  return a + b;
}
template<class T>
T Add(const T a, const T b)
{
  return a + b;
}
void Test2()
{
  int a = 10, b = 20;
  Add(a, b);//与非模板函数匹配,编译器不需要特化
  Add<int>(a, b);//调用编译器特化的Add版本
}


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

int Add(int a, int b)
{
  return a + b;
}
template<class T1, class T2>
T1 Add(const T1 a, const T2 b)
{
  return a + b;
}
void Test3()
{
  int a = 10, b = 20;
  double c = 1.1;
  Add(a, b);
  Add(a, c);
}


ce06dbe8025181c52763efc1bf5ceb8a.png

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


3. 类模板


类模板和函数模板非常相似

1.类模板的定义格式

template<class T1, class T2, ......>
class 模板名
{
  //类内成员定义  
};


例如,vector类模板

template<class T>
class vector//注意,这里的vector不是具体的类,是编译器根据被实例化的类型生成具体类的模具
{
public:
  vector()
    :_start(nullptr)
    , _finish(nullptr)
    , _endOfStorage(nullptr)
  {}
  ~vector()
  {
    delete[] _start;
    _start = _finish = _endOfStorage = nullptr;
  }
  //......
private:
  T* _start;
  T* _finish;
  T* _endOfStorage;
};


2.类模板的实例化

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

//vector
Test4()
{
    vector<int> vi;
    vector<string> vstr;
}


相关文章
|
3月前
|
存储 算法 C++
C++ STL 初探:打开标准模板库的大门
C++ STL 初探:打开标准模板库的大门
136 10
|
5月前
|
编译器 C++
【C++】——初识模板
【C++】——初识模板
【C++】——初识模板
|
6月前
|
程序员 C++
C++模板元编程入门
【7月更文挑战第9天】C++模板元编程是一项强大而复杂的技术,它允许程序员在编译时进行复杂的计算和操作,从而提高了程序的性能和灵活性。然而,模板元编程的复杂性和抽象性也使其难以掌握和应用。通过本文的介绍,希望能够帮助你初步了解C++模板元编程的基本概念和技术要点,为进一步深入学习和应用打下坚实的基础。在实际开发中,合理运用模板元编程技术,可以极大地提升程序的性能和可维护性。
|
2月前
|
安全 编译器 C++
【C++11】可变模板参数详解
本文详细介绍了C++11引入的可变模板参数,这是一种允许模板接受任意数量和类型参数的强大工具。文章从基本概念入手,讲解了可变模板参数的语法、参数包的展开方法,以及如何结合递归调用、折叠表达式等技术实现高效编程。通过具体示例,如打印任意数量参数、类型安全的`printf`替代方案等,展示了其在实际开发中的应用。最后,文章讨论了性能优化策略和常见问题,帮助读者更好地理解和使用这一高级C++特性。
85 4
|
2月前
|
算法 编译器 C++
【C++】模板详细讲解(含反向迭代器)
C++模板是泛型编程的核心,允许编写与类型无关的代码,提高代码复用性和灵活性。模板分为函数模板和类模板,支持隐式和显式实例化,以及特化(全特化和偏特化)。C++标准库广泛使用模板,如容器、迭代器、算法和函数对象等,以支持高效、灵活的编程。反向迭代器通过对正向迭代器的封装,实现了逆序遍历的功能。
39 3
|
2月前
|
编译器 C++
【c++】模板详解(1)
本文介绍了C++中的模板概念,包括函数模板和类模板,强调了模板作为泛型编程基础的重要性。函数模板允许创建类型无关的函数,类模板则能根据不同的类型生成不同的类。文章通过具体示例详细解释了模板的定义、实例化及匹配原则,帮助读者理解模板机制,为学习STL打下基础。
39 0
|
3月前
|
编译器 程序员 C++
【C++打怪之路Lv7】-- 模板初阶
【C++打怪之路Lv7】-- 模板初阶
28 1
|
3月前
|
存储 编译器 C++
【C++篇】引领C++模板初体验:泛型编程的力量与妙用
【C++篇】引领C++模板初体验:泛型编程的力量与妙用
59 9
|
3月前
|
编译器 C语言 C++
C++入门6——模板(泛型编程、函数模板、类模板)
C++入门6——模板(泛型编程、函数模板、类模板)
79 0
C++入门6——模板(泛型编程、函数模板、类模板)
|
3月前
|
算法 编译器 C++
【C++篇】领略模板编程的进阶之美:参数巧思与编译的智慧
【C++篇】领略模板编程的进阶之美:参数巧思与编译的智慧
104 2