C++:模板的相关内容

简介: C++:模板的相关内容

本篇介绍一部分关于C++中模板使用的问题,模板是C++的一大特色,需要在实际运用中体会它的妙处

泛型编程

为了知道什么是泛型编程,先来看,如何实现对于所有类型都使用的交换函数?

在C语言中这个实现是不可以的,因为没有函数重载,在C++中引入了函数重载的概念,根据函数参数的类型不同在寻找符号表的时候函数名不同,因此构成了函数重载,即使使用函数重载可以解决这个问题,函数重载依旧有很多弊端,比如同样的代码语句逻辑一摸一样,只有最后的函数参数的类型和函数体中的类型不一样,因此这里引入了模板的概念:

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

因此就有了函数模板的概念

函数模板

格式:

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

因此swap函数就可以写成

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

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

函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。所以其实模

板就是将本来应该我们做的重复的事情交给了编译器

那实际函数调用的时候,编译器调用的是同一个函数吗,其实并不是这样

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

类模板

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

非类型模板参数

模板参数主要有两种,一种是类型形参,另外一种是非类型形参

  • 类型形参:出现在模板参数列表中,也就是在class或者typename后面的参数类型名称就是类型形参
  • 非类型形参:用一个常量作为类模板的一个参数,在类模板中可以将该参数作为常量来使用
namespace static_array
{
  // 定义一个模板类型的静态数组
  template<class T, size_t N = 10>
  class array
  {
  public:
    T& operator[](size_t index) 
    { 
      return _array[index]; 
    }
    const T& operator[](size_t index) const 
    { 
      return _array[index]; 
    }
    size_t size() const 
    { 
      return _size; 
    }
    bool empty() const 
    { 
      return 0 == _size; 
    }
  private:
    T _array[N];
    size_t _size;
  };
}

以上面的为例,通过这样的非类型模板参数可以实现定义一个常量的功能

值得注意的有下面两点:

  1. 浮点数,类对象和字符串不能作为非类型模板参数
  2. 非类型模板参数必须在编译前就确定,因为编译器要根据这个值开辟空间等操作

模板的特化

什么是模板的特化?

模板可以缩减代码的重复性,也就是可以实现一些和类型不相关的代码,但是也会有特殊的情况,比如说对于一些特殊的类型就会得到一些错误的结果,如果遇到这种情况该如何处理呢?

C++就引入了特化的概念,对于特殊的情况可以采用特殊的结果进行处理,那么有什么场景可以对这些进行使用?

以下面的例子为例:

#include <bits/stdc++.h>
using namespace std;
template <class T>
bool Less(T a, T b)
{
  return a < b;
}
int main()
{
  int p = 2;
  int q = 1;
  cout << Less(1, 2) << endl;
  cout << Less(1.1, 2.2) << endl;
  cout << Less(&p, &q) << endl;
  return 0;
}

运行结果如下

问题此时就出现了,对于模板来说,它可以比较各种类型的值,但是对于指针类型来说,它只能把指针的地址当成一个值来进行比较,因此是无法得到正确的结果的,此时可以用特化的原理来解决问题:

#include <bits/stdc++.h>
using namespace std;
template <class T>
bool Less(T a, T b)
{
  return a < b;
}
template <>
bool Less<int*>(int* a, int* b)
{
  return (*a) < (*b);
}
int main()
{
  int p = 2;
  int q = 1;
  cout << Less(1, 2) << endl;
  cout << Less(1.1, 2.2) << endl;
  cout << Less(&p, &q) << endl;
  return 0;
}

运行结果如下

上面的写法就是函数模板的特化,有下面一些需要注意的步骤

  • 必须有一个基础的函数模板
  • 关键字template后要带一个尖括号
  • 函数名后要带尖括号,内容是要被特化的类型
  • 函数的参数表要和模板函数的基础参数类型完全相同,才能实现函数模板的特化

注意,一般不使用函数特化,为了实现简单都是直接将函数给出,这样更加便携

#include <bits/stdc++.h>
using namespace std;
template <class T>
bool Less(T a, T b)
{
  return a < b;
}
// 直接给出函数,这样阅读起来更加便捷,可读性更高
bool Less(int* a, int* b)
{
  return (*a) < (*b);
}
int main()
{
  int p = 2;
  int q = 1;
  cout << Less(1, 2) << endl;
  cout << Less(1.1, 2.2) << endl;
  cout << Less(&p, &q) << endl;
  return 0;
}


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