【C++】初识C++模板与STL

简介: 【C++】初识C++模板与STL

一、泛型编程

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

具体场景:实现一个通用的交换函数

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;
}
void Swap(char& left, char& right)
{
    char temp = left;
    left = right;
    right = temp;
}
.....................

通过函数重载实现通用函数缺陷:

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

如果在C++中,存在一个摸具,通过给这个摸具中填充不同材料(类型),赖获得不同材料的锻件(即生成具体类型的代码),那么就会节省许多头发。对此C++提出模板的概念,对于模板分为函数模板以及类模板。

二、函数模板

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

函数模板格式:template

使用函数模板实现通用交换函数

template<typename T>
void Swap(T& left, T& right)
{
  T temp = left;
  left = right;
  right = temp;
}
int main()
{
  int a = 10; int b = 20;
  cout << a << "/" << b<<endl;
  Swap(a, b);
  cout << a << "/" << b << endl;
  
  double c = 1.1; double d = 2.2;
  cout << c << "/" << d << endl;
  Swap(c, d);
  cout << c << "/" << d << endl;
  return 0;
}

其中支持typenameclass来定义模板参数关键字,但是不能使用struct来代替class定义。

提出思考:当我们编写了函数模板,两次函数调用是否为同一函数呢?

:调用不是同一个函数,虽然在调试中都执行到模板函数体中,但是这只是编译器为了方便观察进行的调整。对于不同类型所占用空间大小不是相同以及浮点数存储和释放都有自己的规定。(Swap函数在库实现好了并且C++有模块的概念,可以直接调用库中Swap函数)

2.1 函数模板原理

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

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

2.2 函数模板实例化

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

如图该语句不能通过编译器,由于编译期间,编译器进行实例化需要推演其实参类型。

报错理由:通过实参a1将T推演为int,通过实参d1将T推演为double类型,但是模板参数列表中只有一个T,编译器无法确定此处到底该将T确定为int或者double类型而报错。(在模板中编译器一般不会进行类型转换操作,因为一旦转化出问题,编译器就要背黑锅了。)

那么针对上面出现的问题,有三种解决方案

  • 用户自己来强制转化
  • 使用显式实例化
  • 添加一个T2

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

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

第一种:强制转化

int main()
{
  int a1 = 10,a2 = 20;
  double d1 = 10.2, d2 = 20.2;
  Add(a1, (int)d1);
  Add((double)a1, d2);
  return 0;
}

第二种:显式实例化

int main()
{
  int a1 = 10,a2 = 20;
  double d1 = 10.2, d2 = 20.2;
  Add<int>(a1, d1);
  Add<double>(a1, d1);  
  return 0;
}

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

第三种:添加一个T2(这里返回值的类型,需要用户选择T1 or T2)

template<class T1,class T2>
T1 Add(const T1& left, const T2& right)
{
  cout << left + right<< endl;
  return left + right;
}
int main()
{
  int a1 = 10,a2 = 20;
  double d1 = 10.2, d2 = 20.2;
  Add(a1, d1);
  Add(a1, d1);
  
  return 0;
}

2.3 模板参数匹配原则

当同名函数模板与非模板函数同时存在,该函数模板可以实例化为非函数模板。

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

如果同名函数模板与非函数模板,并且其他条件相同,在调用时会优先调用非函数模板(有现成的吃现成的菜)而不会从该模板实例化一个,除非模板可以产生一个具有更好匹配的函数,在调用时优先选择模板(现成的不好吃,不如吃自己做的)就像是想我委屈嫁给你,不如我找个有钱的大爷~

templace<class T1, class T2>
T1 Add(T1 left, T2 right)
{
  return left + right;
}
int main()
{
   int ret = Add(1,2.0);
    return 0;
}

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

三、类模板

3.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;
}
int main()
{
    //显示模板实例化
  Vector<int> d1(10);
        vector<double> d2(10.0);
}

类模板可以处理需要很多类型的数据,如果按照C语言那一套TypeData typename使用,当创建不同类型数据,需要修改名字连同实现逻辑也需要更换名字。不如使用模板,将我们需要重复做的事情交给编译器来做。

3.2 类模板的实现化

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

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

四、函数模板与类模板区别

类模板与函数模板不同,编译器不会对类模板进行类型推导,因此在使用类模板时,必须显式指定模板参数。由于类模板可以包含多个成员变量、方法和构造函数,推导类模板的类型比推导函数模板的类型复杂得多。类模板的构造函数可以接受与类的模板参数不同类型的参数(通过转换),这种情况无法准确推导出模板参数。

template <typename T1, typename T2>
struct MyPair {
    MyPair(const T1& first, const T2& second) {
        // 构造函数可以接收其他类型
    }
};
MyPair p(1, "Hello");  // 编译器不知道模板参数类型
  • 函数模板:支持类型推导,编译器根据传递的参数自动推导模板参数。
  • 类模板:不支持类型推导,必须显式指定模板参数类型,因为类的结构和使用方式更复杂,无法通过简单的参数推导出类型。

五、简单了解STL

STL(standard tmplate libaray-标准库):是C++标准库的重要组成部分,不仅是一个可复用的组件库,而且是一个包罗数据结构与算法的软件框架。

5.1 STL的版本

原始版本

Alexander Stepanov、Meng Lee 在惠普实验室完成的原始版本,本着开源精神,他们声明允许任何人任意运用、拷贝、修改、传播、商业使用这些代码,无需付费。唯一的条件就是也需要向原始版本一样做开源使

用。 HP 版本–所有STL实现版本的始祖。

P. J.版本:

由P. J. Plauger开发,继承自HP版本,被Windows Visual C++采用,不能公开或修改,缺陷:可读性比较低,符号命名比较怪异。

RW 版本:

由Rouge Wage公司开发,继承自HP版本,被C+ + Builder 采用,不能公开或修改,可读性一般。

SGI版本:

由Silicon Graphics Computer Systems,Inc公司开发,继承自HP版 本。被GCC(Linux)采用,可移植性好,可公开、修改甚至贩卖,从命名风格和编程 风格上看,阅读性非常高。我们后面学习STL要阅读部分源代码,

主要参考的就是这个版本

5.2 STL的六大组件

5.3 如何学习STL

5.4 STL的缺陷:

  1. STL库的更新太慢了。这个得严重吐槽,上一版靠谱是C++98,中间的C++03基本一些修订。C++11出来已经相隔了13年,STL才进一步更新
  2. STL现在都没有支持线程安全。并发环境下需要我们自己加锁。且锁的粒度是比较大的。
  3. STL极度的追求效率,导致内部比较复杂。比如类型萃取,迭代器萃取
  4. STL的使用会有代码膨胀的问题,比如使用vector/vector/vector这样会生成多份代码,当然这是模板语法本身导致的
  5. 在不久的将来,将更加深入的学习模板的进阶知识。

以上就是本篇文章的所有内容,在此感谢大家的观看!这里是店小二呀C++笔记,希望对你在学习C++语言旅途中有所帮助!


相关文章
|
28天前
|
缓存 算法 程序员
C++STL底层原理:探秘标准模板库的内部机制
🌟蒋星熠Jaxonic带你深入STL底层:从容器内存管理到红黑树、哈希表,剖析迭代器、算法与分配器核心机制,揭秘C++标准库的高效设计哲学与性能优化实践。
C++STL底层原理:探秘标准模板库的内部机制
|
7月前
|
编译器 C++ 容器
【c++丨STL】基于红黑树模拟实现set和map(附源码)
本文基于红黑树的实现,模拟了STL中的`set`和`map`容器。通过封装同一棵红黑树并进行适配修改,实现了两种容器的功能。主要步骤包括:1) 修改红黑树节点结构以支持不同数据类型;2) 使用仿函数适配键值比较逻辑;3) 实现双向迭代器支持遍历操作;4) 封装`insert`、`find`等接口,并为`map`实现`operator[]`。最终,通过测试代码验证了功能的正确性。此实现减少了代码冗余,展示了模板与仿函数的强大灵活性。
196 2
|
7月前
|
存储 算法 C++
【c++丨STL】map/multimap的使用
本文详细介绍了STL关联式容器中的`map`和`multimap`的使用方法。`map`基于红黑树实现,内部元素按键自动升序排列,存储键值对,支持通过键访问或修改值;而`multimap`允许存在重复键。文章从构造函数、迭代器、容量接口、元素访问接口、增删操作到其他操作接口全面解析了`map`的功能,并通过实例演示了如何用`map`统计字符串数组中各元素的出现次数。最后对比了`map`与`set`的区别,强调了`map`在处理键值关系时的优势。
399 73
|
4月前
|
存储 算法 安全
c++模板进阶操作——非类型模板参数、模板的特化以及模板的分离编译
在 C++ 中,仿函数(Functor)是指重载了函数调用运算符()的对象。仿函数可以像普通函数一样被调用,但它们实际上是对象,可以携带状态并具有更多功能。与普通函数相比,仿函数具有更强的灵活性和可扩展性。仿函数通常通过定义一个包含operator()的类来实现。public:// 重载函数调用运算符Add add;// 创建 Add 类的对象// 使用仿函数return 0;
146 0
|
4月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
116 0
|
8月前
|
存储 缓存 C++
C++ 容器全面剖析:掌握 STL 的奥秘,从入门到高效编程
C++ 标准模板库(STL)提供了一组功能强大的容器类,用于存储和操作数据集合。不同的容器具有独特的特性和应用场景,因此选择合适的容器对于程序的性能和代码的可读性至关重要。对于刚接触 C++ 的开发者来说,了解这些容器的基础知识以及它们的特点是迈向高效编程的重要一步。本文将详细介绍 C++ 常用的容器,包括序列容器(`std::vector`、`std::array`、`std::list`、`std::deque`)、关联容器(`std::set`、`std::map`)和无序容器(`std::unordered_set`、`std::unordered_map`),全面解析它们的特点、用法
C++ 容器全面剖析:掌握 STL 的奥秘,从入门到高效编程
|
7月前
|
编译器 C++
模板(C++)
本内容主要讲解了C++中的函数模板与类模板。函数模板是一个与类型无关的函数家族,使用时根据实参类型生成特定版本,其定义可用`typename`或`class`作为关键字。函数模板实例化分为隐式和显式,前者由编译器推导类型,后者手动指定类型。同时,非模板函数优先于同名模板函数调用,且模板函数不支持自动类型转换。类模板则通过在类名后加`&lt;&gt;`指定类型实例化,生成具体类。最后,语录鼓励大家继续努力,技术不断进步!
|
7月前
|
存储 算法 C++
【c++丨STL】set/multiset的使用
本文深入解析了STL中的`set`和`multiset`容器,二者均为关联式容器,底层基于红黑树实现。`set`支持唯一性元素存储并自动排序,适用于高效查找场景;`multiset`允许重复元素。两者均具备O(logN)的插入、删除与查找复杂度。文章详细介绍了构造函数、迭代器、容量接口、增删操作(如`insert`、`erase`)、查找统计(如`find`、`count`)及`multiset`特有的区间操作(如`lower_bound`、`upper_bound`、`equal_range`)。最后预告了`map`容器的学习,其作为键值对存储的关联式容器,同样基于红黑树,具有高效操作特性。
308 3
|
8月前
|
存储 算法 C++
【c++丨STL】priority_queue(优先级队列)的使用与模拟实现
本文介绍了STL中的容器适配器`priority_queue`(优先级队列)。`priority_queue`根据严格的弱排序标准设计,确保其第一个元素始终是最大元素。它底层使用堆结构实现,支持大堆和小堆,默认为大堆。常用操作包括构造函数、`empty`、`size`、`top`、`push`、`pop`和`swap`等。我们还模拟实现了`priority_queue`,通过仿函数控制堆的类型,并调用封装容器的接口实现功能。最后,感谢大家的支持与关注。
455 1
|
8月前
|
安全 C++
【c++】模板详解(2)
本文深入探讨了C++模板的高级特性,包括非类型模板参数、模板特化和模板分离编译。通过具体代码示例,详细讲解了非类型参数的应用场景及其限制,函数模板和类模板的特化方式,以及分离编译时可能出现的链接错误及解决方案。最后总结了模板的优点如提高代码复用性和类型安全,以及缺点如增加编译时间和代码复杂度。通过本文的学习,读者可以进一步加深对C++模板的理解并灵活应用于实际编程中。
117 0