C++模板初阶

简介: C++模板初阶

大家在做OJ题的时候可能看到过这种使用STL的样子:

其实这个就是我们今天要介绍的模板这个语法在STL中的应用

一.泛型编程

在介绍泛型编程的概念之前,先给大家看一下这个例子

比方说:我今天要分别实现int和int类型,double和double类型,char和char类型的Swap函数

那么我可能需要利用函数重载写出一下三份代码

void Swap(int& a, int& b)
{
   int tmp = a;
   a = b;
   b = tmp;
}
void Swap(double& a, double& b)
{
   double tmp = a;
   a = b;
   b = tmp;
}
void Swap(char& a, char& b)
{
   char tmp = a;
   a = b;
   b = tmp;;
}

这样也是可以做到的

但是这样好不好呢?

二.函数模板

1.概念

2.实例

因此,我们就可以这样优化那份Swap函数了

//typename是用来定义模板参数关键字,也可以使用class(切记:不能使用struct代替class)
//template<typename T>//T:type
template<class T>//T:type
void Swap(T& a, T& b)
{
  T tmp = a;
  a = b;
  b = tmp;
}
int main()
{
  int a = 0, b = 1;
  Swap(a, b);//模板的实例化
  double c = 2.1, d = 3.1;
  Swap(c, d);
  char e = 'a', f = 'b';
  Swap(e, f);
  return 0;
}

下面问题来了,这三次调用Swap函数调用的是同一个函数吗?

答案是:并不是,为什么呢?

这就涉及到下面的函数模板的原理了

3.原理

因此,如果我的这个函数调用让编译器无法推演出T的类型,也就是下面这种情况,编译器就会报错

(这里我使用Add这个函数来为下面做例子,关于为什么不用Swap函数我会在介绍的时候一并说明的)

注意:在模板中,编译器一般不会进行隐式类型转换操作,

因为一旦转化出问题,编译器就需要背黑锅

能不能通过某种手段来解决这种错误呢?

可以使用下面介绍的函数模板的实例化

4.函数模板的实例化

1.隐式实例化

我们可以这样去做:

这里我通过强制类型转换

分别将i和d这两个变量转换为int和double类型,成功完成了相加

但是:

那么为什么Swap函数就不可以呢?

我们之前在C++入门-引用中提到过:

那么接下来我们来看第二种方法:显式实例化

2.显式实例化

这里里面的int就是告诉编译器,这个T的类型就是int,同理也是如此

那么这种方法能不能解决Swap呢?

当然不行啦,因为我这个Swap的报错是权限放大导致的报错,不是T的类型无法推演出来所导致的报错

注意:

1.当我们无法通过传参的方式来让函数模板推演出T的类型的时候,才是显式实例化真正的用武之地:

2.模板参数列表也可以定义很多T

5.模板参数的匹配原则

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

2. 对于非模板函数和同名函数模板:

如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例

如果模板可以产生一个具有更好匹配的函数 , 那么将选择模板

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

三.类模板

1.类模板的引出

我们在学习数据结构的时候经常会用到

typedef int STDateType;

等等这样的typedef

但是typedef也不能解决所有问题:

比如这种问题:

如果我今天定义两个栈对象,一个存int,一个存double,这种情况下typedef就无法解决了,

那么该怎么办呢?

这个时候就要用到类模板了

2.实例

语法形式跟函数模板类似

不过:

也就是说这里的真正的类是

Stack Stack Stack Stack

3.易错点:类模板的声明跟定义分离

template<class T>
class Stack
{
public:
  Stack(int capacity = 4);
  ~Stack();
  //注意:构造函数不能这样写:Stack<T>{...}
  void Push(const T& val);
private:
  T* _a;
  int _top;
  int _capacity;
};
//类模板的声明和定义分离:必须这么分离
//但是类模板的声明和定义分离必须放到同一个文件当中
//不能说是:类模板定义的类声明放在.h中,定义放在.cpp中,否则会报错的!
template<class T>
Stack<T>::Stack(int capacity)//这里指定类域要加上T,而且前面的template<class T>也要加上
  :_top(0)
  , _capacity(capacity)
{
  _a = new T[capacity];
  cout << "Stack()" << endl;
}
template<class T>
Stack<T>::~Stack()
{
  delete[]_a;
  _a = nullptr;
  _top = _capacity = 0;
  cout << "~Stack()" << endl;
}
template<class T>
void Stack<T>::Push(const T& val)
{
  //...
}

以上就是C++模板初阶的全部内容,希望能对大家有所帮助!

相关文章
|
3月前
|
缓存 算法 程序员
C++STL底层原理:探秘标准模板库的内部机制
🌟蒋星熠Jaxonic带你深入STL底层:从容器内存管理到红黑树、哈希表,剖析迭代器、算法与分配器核心机制,揭秘C++标准库的高效设计哲学与性能优化实践。
C++STL底层原理:探秘标准模板库的内部机制
|
7月前
|
存储 算法 安全
c++模板进阶操作——非类型模板参数、模板的特化以及模板的分离编译
在 C++ 中,仿函数(Functor)是指重载了函数调用运算符()的对象。仿函数可以像普通函数一样被调用,但它们实际上是对象,可以携带状态并具有更多功能。与普通函数相比,仿函数具有更强的灵活性和可扩展性。仿函数通常通过定义一个包含operator()的类来实现。public:// 重载函数调用运算符Add add;// 创建 Add 类的对象// 使用仿函数return 0;
245 0
|
7月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
194 0
|
10月前
|
编译器 C++
模板(C++)
本内容主要讲解了C++中的函数模板与类模板。函数模板是一个与类型无关的函数家族,使用时根据实参类型生成特定版本,其定义可用`typename`或`class`作为关键字。函数模板实例化分为隐式和显式,前者由编译器推导类型,后者手动指定类型。同时,非模板函数优先于同名模板函数调用,且模板函数不支持自动类型转换。类模板则通过在类名后加`&lt;&gt;`指定类型实例化,生成具体类。最后,语录鼓励大家继续努力,技术不断进步!
|
11月前
|
编译器 C++
㉿㉿㉿c++模板的初阶(通俗易懂简化版)㉿㉿㉿
㉿㉿㉿c++模板的初阶(通俗易懂简化版)㉿㉿㉿
|
11月前
|
安全 C++
【c++】模板详解(2)
本文深入探讨了C++模板的高级特性,包括非类型模板参数、模板特化和模板分离编译。通过具体代码示例,详细讲解了非类型参数的应用场景及其限制,函数模板和类模板的特化方式,以及分离编译时可能出现的链接错误及解决方案。最后总结了模板的优点如提高代码复用性和类型安全,以及缺点如增加编译时间和代码复杂度。通过本文的学习,读者可以进一步加深对C++模板的理解并灵活应用于实际编程中。
165 0
|
11月前
|
存储 安全 算法
深入理解C++模板编程:从基础到进阶
在C++编程中,模板是实现泛型编程的关键工具。模板使得代码能够适用于不同的数据类型,极大地提升了代码复用性、灵活性和可维护性。本文将深入探讨模板编程的基础知识,包括函数模板和类模板的定义、使用、以及它们的实例化和匹配规则。
|
安全 编译器 C++
【C++11】可变模板参数详解
本文详细介绍了C++11引入的可变模板参数,这是一种允许模板接受任意数量和类型参数的强大工具。文章从基本概念入手,讲解了可变模板参数的语法、参数包的展开方法,以及如何结合递归调用、折叠表达式等技术实现高效编程。通过具体示例,如打印任意数量参数、类型安全的`printf`替代方案等,展示了其在实际开发中的应用。最后,文章讨论了性能优化策略和常见问题,帮助读者更好地理解和使用这一高级C++特性。
411 4
|
算法 编译器 C++
【C++】模板详细讲解(含反向迭代器)
C++模板是泛型编程的核心,允许编写与类型无关的代码,提高代码复用性和灵活性。模板分为函数模板和类模板,支持隐式和显式实例化,以及特化(全特化和偏特化)。C++标准库广泛使用模板,如容器、迭代器、算法和函数对象等,以支持高效、灵活的编程。反向迭代器通过对正向迭代器的封装,实现了逆序遍历的功能。
202 3
|
编译器 C++
【c++】模板详解(1)
本文介绍了C++中的模板概念,包括函数模板和类模板,强调了模板作为泛型编程基础的重要性。函数模板允许创建类型无关的函数,类模板则能根据不同的类型生成不同的类。文章通过具体示例详细解释了模板的定义、实例化及匹配原则,帮助读者理解模板机制,为学习STL打下基础。
180 0