STL体系结构概述

文章目录

    • STL是什么?
    • STL的六大组件
    • STL的实现版本
    • 额外补充
      • 一、容器范围区间
      • 二、容器结构与分类
        • 序列式容器
        • 关联容器
          • 有序关联容器
          • 不定序关联容器
    • 参考

本文将是STL系列的第一篇文章,主要参考《STL源码剖析》,辅以网络博文,不定时更新我感兴趣的内容。

笔者在入门C++这门编程语言的时候之前,就已听腻了这个STL这个词了,特别是在OJ场景下,主要是因为采用这个库就可以避免手撸一些数据结构,有相关数据结构基础就能直接上手了。当然,这都是对STL比较浅显的认知罢了,本系列将开始学习STL的相关知识。

STL是什么?

STLStandard Template Library的缩写,中文译作标准模板库,属于C++标准库的一部分。

STL所实现的,是依据泛型思想架设起来的一个概念结构。这个以抽象概念(abstract concepts)为主体而非以实际类(classes)为主体的结构,形成了一个严谨的接口标准。在此接口之下,任何组件都有最大的独立性,并以所谓的迭代器(iterator)胶合起来,或以适配器(adapter)互相适配,或以所谓仿函数(functor)动态选择某种策略(policy或strategy)。

STL的六大组件

STL提供六大组件,彼此之间相辅相成:

  • 容器(containers):各种数据结构,如vectorlistdequemapset用来存储数据。从实现上来说,STL容器是class template,也就是说它能够支持不同类型的元素(前提是需要满足容器要求)。
  • 算法(algorithms):各种常用算法,如sortsearchcopy等。从实现上来说,它是一种function template
  • 迭代器(iterators):扮演容器与算法之间的胶合剂,是一种“泛型指针”,也可以理解成抽象化的指针,实现上是借助了C++中的操作符重载的特性,所有的STL容器都附带了一套属于自己的专属迭代器。原生指针(native pointer)也是一种迭代器。
  • 仿函数(functiors):行为类似于函数,可作为算法的某种策略(policy)。从实现上来说,其实就是一种重载了operator ()classclass template,比如lambda表达式也是其中一种。
  • 配接器(adapters):适配器,一种用来修饰容器(containter)或仿函数(functiors)或迭代器接口的东西。比如,STL中提供的queuestack是一种容器适配器,底层默认是deque容器。改变functor者,称之为function adapter;改变container接口者,称为container adapter;改变iterator接口者,称为iterator adapter
  • 配置器(allocators):分配器,负责空间的配置和管理。从实现上看,它是一个实现了动态空间配置、空间管理、空间释放的class template

它们之间的关系可以用下图进行表示:

在这里插入图片描述

要理解这张图,需要读者结合上面对六大组件的简单解释来分析,如果还有一些STL使用经历就更好了。

STL的实现版本

在学习C++过程,大家需要理解一个概念就是:我们学习的是一个标准(C++11、C++14、C++17等标准),而具体实现依据不同编译器,标准没规定的就是由编译器自由发挥了。在标准中没规定的东西,不要从某一个编译器实现中看到它的实现方式就说C++对这个一定是这样实现的,这是一种思维误区,这是不对的。

STL实现主要有以下的版本:

  • HP实现的版本:所有STL实现的始祖
  • P.J.Plauger实现的版本:继承于HP版本,被**Visual C++**采用
  • Rouge Wave实现的版本:继承于HP版本,被C++Builder采用
  • STLPort实现的版本:提供一个以SGI STL为蓝本的高度可移植性实现的版本
  • SGI STL实现的版本:同样继承自HP版本,被GCC采用

在侯捷的《STL源码剖析》一书中分析的便是SGI STL实现的版本。一方面,GCCC++的支持性比较高,使用率也比较高,大家平常接触得比较多的也应该是这个版本,分析这个版本的内容,也更好了解GCC;另一方面,SGI STL版本可读性很高,值得学习。

额外补充

一、容器范围区间

容器提供的范围区间一般是遵循前闭后开[)的原则。典型地,就是来自begin()end()函数返回的这两个迭代器所指代的范围:

在这里插入图片描述

begin()返回的迭代器指向了容器元素中的第一个,而end()返回的迭代器则是容器元素中的最后一个的下一个(不存在的元素)。这是一个例子,其他的返回区间,也是遵循这一原则。

二、容器结构与分类

容器主要可以分为两大类:

  • 序列式容器(SequenceContainer):在线性排列中存储相同类型对象的容器
  • 关联容器(AssociativeContainer):提供基于键的快速对象查找的容器
序列式容器

序列式容器底层是采用类似线性表的结构来存储元素

  1. array:于C++11标准引入,功能等同于传统数组,但它不会退化为指针,还能提供容器相关的操作
  2. vector:封装动态数组的序列容器,特点是 存储动态管理,尾部扩增,在GCC中,其增长空间是以2倍增长的
  3. deque:双端队列,头尾两端可增可减,内部存储的方式是以分段模拟连续空间(关键靠迭代器),访问元素需要两层指针访问。
  4. list:通常实现为双向链表,支持从容器任何位置进行常数时间的元素插入和移除,不支持随机访问。
  5. forward_list:于C++11标准引入,通常实现为单链表,支持从容器中的任何位置快速插入和移除元素的容器,不支持随机访问。

序列式容器不止上述这些,随着新标准的引入,相信也会有新的容器加入到STL中,建议关注:cppreference.com

关联容器
  • 有序关联容器:底层通常采用红黑树实现
  • 不定序关联容器:底层通常采用哈希表哈希表实现
有序关联容器
  1. set:只存储Key类型的对象,有序,Key唯一
  2. multiset:只存储Key类型的对象,有序,Key允许重复
  3. map:存储键值对,有序,Key唯一
  4. multimap:存储键值对,有序,Key允许重复
不定序关联容器
  1. unordered_set:只存储Key类型的对象,无特别的顺序,只是组织到桶中,依赖hash值,Key唯一
  2. unordered_multiset:只存储Key类型的对象,无特别的顺序,只是组织到桶中,依赖hash值,Key允许重复
  3. unordered_map:存储键值对,无特别顺序,只是组织到桶中,依赖hash值,Key唯一
  4. unordered_multimap:存储键值对,无特别顺序,只是组织到桶中,依赖hash值,Key允许重复

当然,关联式容器也不止这些,这里只是简单做个介绍。

参考

  • 容器库 - cppreference.com
  • STL源码剖析 (豆瓣) (douban.com)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/520938.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

我所理解的工程师文化

对于什么是工程师文化,很多人都能说出很多关键词,比如:匠心,创新,专业,担当,开放…… 然而如果要用一句话来描述什么是工程师文化,如何判断一家公司是不是一家工程师文化的公司&…

matlab中的libsvm怎么录入数据啊,LibSVM在MATLAB中使用时的几个问题

在科研中需要用到支持向量机(Support Vector Machines, SVM)来进行分类,而目前比较成熟的用于实现SVM的软件包则首推LibSVM。LibSVM目前的版本已经能直接在MATLAB中使用了,虽然MATLAB已经提供了诸如svmtrain等SVM相关的函数,可是相比LibSVM的…

JAVA实现AES 解密报错Input length must be multiple of 16 when decrypting with padded cipher

加密代码 /*** 加密* * param content 需要加密的内容* param password 加密密码* return*/public static byte[] encrypt(String content, String password) {try { KeyGenerator kgen KeyGenerator.getInstance("AES");kgen.init(128, new SecureRando…

python 判断时间是否大于6点_python中判断时间间隔的问题

展开全部代码有点烂,不过还算能解决问题注释比较详细了# -*- coding: utf-8 -*-import datetime__author__ lpe234__date__ 2015-04-26f file(1.txt)file_content f.readlines()all_lines len(file_content)def get_(content):"""递归调用:para…

迁移学习让深度学习更容易

深度学习在一些传统方法难以处理的领域有了很大的进展。这种成功是由于改变了传统机器学习的几个出发点,使其在应用于非结构化数据时性能很好。如今深度学习模型可以玩游戏,检测癌症,和人类交谈,自动驾驶。 深度学习变得强大的同…

java多线程系列:通过对战游戏学习CyclicBarrier

CyclicBarrier是java.util.concurrent包下面的一个工具类,字面意思是可循环使用(Cyclic)的屏障(Barrier),通过它可以实现让一组线程到达一个屏障(也可以叫同步点)时被阻塞&#xff0…

Wi-Fi 6到底有什么特别?

戳蓝字“CSDN云计算”关注我们哦!作者 | 甜草莓责编 | 阿秃▋ 什么是Wi-Fi 6 ?Wi-Fi 6,是Wi-Fi联盟给IEEE Std. P802.11ax起的别名。众所周知,以前我们的Wi-Fi都是叫作802.11a/b/n/g/ac/ax之类的名字。这种命名方式实在容易让人…

matlab直流电机pid调速仿真,直流电机双闭环PID调速系统仿真设计

目录直流电机双闭环PID调速系统仿真1 转速、电流双闭环直流调速系统的组成及工作原理2 双闭环调速系统的动态数学模型3 调节器的设计3.1 电流调节器的设计3.2 转速调节器的设计4 搭建模型5 参数计算5.1 参数的直接计算5仿真具体参数6 仿真结果7 结束语8 参考文献摘要在工程的应…

【小技巧】桌面图标出现蓝色问号的怎么办?

【背景】 今天因为要联调前端所以用SVN更新了一下前端的代码,结果没想到桌面图标全部变成了带着蓝色问号的,吓了自己一跳。查了一下之后顺利解决了,不过以防万一记录一下解决方法。 【内容】 解决步骤:首先确定是SVN的问题导致了…

恒强制版系统980_速来围观 | 恒强制版小图高级功能讲解

点击蓝字关注我们 Spring comes小图是用自定义的动作来定义使用者色码(120-183)。小图的主要构成如下图:1. 模块色数属性模块色数有如下规则:(1) 小于100(一般填1)的小图为普通小图【980修改】;(2) 小于200大于100(一般填101)的小图为提花小图…

读懂这篇文章,你的阿里技术面就可以过关了 | Apache RocketMQ 101

在美国的大学课程中,101是所有课程中的第一门,是新生入学后的必修课程。阿里巴巴中间件技术专家刘振东在上周的Apache RocketMQ开发者沙龙北京站的活动上,进行了主题为《ApacheRocketMQ 101》的分享,帮助开发者从0开始学习 Apache…

HttpClient4.5 简单入门实例(一)

一、所需要的jar包 httpclient-4.5.jar httpcore-4.4.1.jar httpmime-4.5.jar 二、实例 package com.gblfy.test;import java.io.File; import java.io.IOException; import java.net.URL; import java.util.ArrayList; import java.util.List; import java.util…

Nacos 计划发布v0.2版本,进一步融合Dubbo和SpringCloud生态

在近期的Aliware Open Source 成都站的活动上,阿里巴巴高级工程师邢学超(于怀)分享了Nacos v0.2的规划和进度,并对Nacos v0.3的控制台进行了预览。Nacos v0.2将进一步融入Duboo和Spring Cloud生态,帮助开发者更好的在微…

你还在疯狂加班打码?兄dei,不如跟我学做超融合吧!

纵观过去十年,媒体、娱乐、交通、银行、保险、医疗、旅游、物流等行业,无一不打上了数字化的烙印。据统计,一百多年前,公司的平均寿命是67年;而在当今的数字化时代,则锐减至15年。 除此之外,更有…

apache禁止多目录运行php文件下载,Nginx Apache下如何禁止指定目录运行PHP脚本

网站程序的上传目录通常是不需要PHP执行权限,通过限制目录的PHP执行权限可以提网站的安全性,减少被攻击的机率。下面和大家一起分享下如何在Apache和Nginx禁止上传目录里PHP的执行权限。Apache下禁止指定目录运行PHP脚本在虚拟主机配置文件中增加php_fla…

python加载模型包占用内存多大_如何保持Keras模型加载到内存中并在需要时使用它? - python...

我正在阅读Keras blog讲解如何使用Flask创建简单的图像分类器Restful API。我想知道如何在不使用python的其他Web框架中实现加载模型的相同方法。在下面的代码中,将在服务器启动之前将模型加载到内存中,直到服务器处于活动状态,它才会运行&am…

你只差这两步 | 将Sentinel 控制台应用于生产环境

这是围绕 Sentinel 的使用场景、技术对比和实现、开发者实践等维度推出的系列文章的第四篇。 第一篇回顾: Dubbo 的流量防卫兵 | Sentinel如何通过限流实现服务的高可用性 - 传送门 第二篇回顾: RocketMQ 的保险丝| Sentinel 如何通过匀速请求和冷启动…

win10日常操作

C:\Windows\System32\drivers\etc

eclipse分级,分级列表显示 - bieshixuan的个人博客 - OSCHINA - 中文开源技术交流社区...

这是个效果图设计思想是,使用左右两个tableview分别展示NSArray * _allArr;NSMutableArray * _rightArr;UITableView * _leftTableView;UITableView * _rightTableView;初始化_arr [{"全部":[ "棉花", "小麦", "水稻", &q…

python如何实时捕捉cmd显示_如何从Python脚本中捕获Python解释器和/或CMD.EXE的输出? -问答-阿里云开发者社区-阿里云...

如果您正在谈论python解释器或CMD.exe,它是您脚本的“父”,那么不可能。在每个类似POSIX的系统中(现在你正在运行Windows,看起来可能有一些我不知道的怪癖,YMMV)每个进程都有三个流,标准输入,标准输出和标准…