【c++】模板详解(1)

简介: 本文介绍了C++中的模板概念,包括函数模板和类模板,强调了模板作为泛型编程基础的重要性。函数模板允许创建类型无关的函数,类模板则能根据不同的类型生成不同的类。文章通过具体示例详细解释了模板的定义、实例化及匹配原则,帮助读者理解模板机制,为学习STL打下基础。

前言

       当我们学习完c++中类和对象以及动态内存管理的相关知识之后,就可以初步了解STL(标准模板库)并进行学习了。当然,在这之前,有一个关键知识的学习,那就是模板。本篇文章博主就和大家一起探讨模板相关的基础知识。


一、问题引入--泛型编程

       当我们需要针对各种类型的数据实现交换函数时,实现的结果可能是这样的:

void Swap(int& x, int& y)
{
    int t = x;
    x = y;
    y = t;
}
 
void Swap(float& x, float& y)
{
    float t = x;
    x = y;
    y = t;
}
 
void Swap(double& x, double& y)
{
    double t = x;
    x = y;
    y = t;
}

不难发现,这样会造成代码冗余,并且有新类型时,就需要手动增加新的函数。那么有没有办法能够实现一个通用的交换函数呢?


       答案是可以的,实现的方法就是借助模板。模板就像是制作物品的模具,通过向这个模具中填充不同类型的材料,就可以得到不同材料构成的铸件。当我们发现一些程序需要处理不同的类型,但它们的逻辑却是相似的,此时就可以使用模板来创建一个通用的函数或类,需要使用时指定数据类型即可。模板的特性体现了泛型编程的思想,能够大大提高代码的复用率,是泛型编程的基础


模板可以分为函数模板和类模板



接下来我们对这两种模板进行逐一讲解。


二、函数模板

函数模板的概念

函数模板代表了一个函数家族,该函数模板与类型无关,在使用时根据实参的类型产生相应类型的函数。


函数模板的定义格式

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

(下一行紧跟函数定义)

这里的“typename”也可以替换成“class”,但是不可以替换成“struct”。


接下来我们使用函数模板写一个通用的交换函数,并尝试使用它:

#include <iostream>
using namespace std;
 
//函数模板的定义
template<class T>
void Swap(T& x, T& y)
{
    T t = x;
    x = y;
    y = t;
}
 
int main()
{
    int a = 10, b = 20;
    double c = 5.5, d = 6.6;
    char e = 'w', f = 'm';
 
    cout << a << ' ' << b << endl;
    cout << c << ' ' << d << endl;
    cout << e << ' ' << f << endl;
    cout << endl;
 
    Swap(a, b);
    Swap(c, d);
    Swap(e, f);
 
    cout << a << ' ' << b << endl;
    cout << c << ' ' << d << endl;
    cout << e << ' ' << f << endl;
    return 0;
}

运行结果:



函数模板的原理

       函数模板也被称作“模板函数”,但是要注意,函数模板本身是一个“蓝图”、“模具”,而不是一个函数。编译器会识别我们传参的类型,根据“蓝图”来创造出相应的函数。可以说,我们使用函数模板就是将多次编写代码的过程交给了编译器。



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


函数模板的实例化

       当各种类型的参数使用函数模板时,称之为函数模板的实例化。函数模板的实例化可以分为隐式实例化和显示实例化。


隐式实例化

       隐式实例化指的是让编译器根据实参的类型来推演出模板参数的实际类型。举个例子:

template<class T>
T Add(T a, T b)
{
    return a + b;
}
 
int main()
{
    int a = 10, b = 20;
    double c = 1.1, d = 2.2;
 
    Add(a, b);
    Add(c, d);
    return 0;
}

注意,如果让不同类型的两个值相加,就会出现编译报错:

Add(a, c);

因为在编译期间,编译器识别到该实例化时,通过a的类型将T推演为int类型,而通过c的类型将T推演为double类型,但由于模板参数列表当中只有一个T,编译器无法确定T到底是什么类型,就会发生报错。


       此时有两种解决方法:1. 将其中一个参数强制类型转换为与另一个参数相同;2. 使用显示实例化。


显式实例化

       显式实例化指在函数名之后,参数列表之前加一个“< >”,在其中按照顺序指定模板参数的实际类型。举个例子:

template<class T>
T Add(T a, T b)
{
    return a + b;
}
 
int main()
{
    int a = 10, b = 20;
    double c = 1.1, d = 2.2;
 
    Add<int>(a, c);//显式实例化
    return 0;
}

此时如果函数参数的类型不匹配,编译器就会对该参数进行隐式类型转换,就不会出现之前的问题了。


模板参数的匹配原则

       模板参数的匹配原则有如下三点:


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

//非模板函数
int Add(int a, int b)
{
    return a + b;
}
 
//同名的函数模板
template<class T>
T Add(T a, T b)
{
    return a + b;
}
 
int main()
{
    int a = 10, b = 20;
 
    Add(a, b);//此时与非模板函数匹配,直接调用之
    Add<int>(a, b);//使用模板进行实例化
    return 0;
}

2. 当非模板函数与同名函数模板同时存在且模板可以产生一个更匹配的函数时,优先选择模板。如果非模板函数更加匹配,则优先选择函数。例如:

//非模板函数
int Add(int a, int b)
{
    return a + b;
}
 
//同名的函数模板
template<class T>
T Add(T a, T b)
{
    return a + b;
}
 
int main()
{
    Add(1.1, 2.2);//使用模板更加匹配参数类型
    return 0;
}

3. 模板函数不允许隐式类型转换,但普通函数可以。


三、类模板

类模板的定义格式

       与函数模板相同,类模板也可以根据不同的类型产生不同的类。它的定义格式如下:


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

(下一行紧跟类的定义)


举个例子:

//类模板
template<class T>
class A
{
public:
    void fun()
    {
        //...
    }
private:
    T _a;
    T _b;
};

当类中函数的声明和定义分离时,需要在定义处重新定义一次模板参数:

template<class T>
void A<T>::fun()
{
    //...
}

且声明和定义不应分离到两个文件,否则会出现链接错误 。


类模板的实例化

       与函数模板不同,类模板只能显示实例化。类模板的名字“A”只是类标签,不是类名,而实例化的结果(例如A<int>)才是真正的类名。

//类模板的实例化
A<int> a;
A<double> b;

总结

       今天我们学习了c++的模板,它分为函数模板和类模板,是泛型编程和学习STL的基础。如果你觉得博主讲的还不错,就请留下一个小小的赞在走哦,感谢大家的支持❤❤❤

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