简述C++ STL关联式容器

关联式容器概述

关联式容器的每个元素都有一个key值一个value值。当元素被插入到关联式容器时,容器内部结构便依照key值大小,按照某种规则将该元素置于适当位置。

关联式容器没有头尾之分,自然也就没有push_back()、pop_front()等操作。

关联式容器的内部结构只有两种:

  • 以红黑树(RB-tree)为底层结构:
    • set
    • map
    • multiset
    • multimap
  • 以哈希表(Hash-table)为底层结构:
    • unordered_set
    • unordered_map
    • unordered_multiset
    • unordered_multimap

Tree的导览

  • 节点的大小:该节点及其所有子节点的节点总数;
  • 路径长度:A节点到B节点的唯一路径经过的边数;
  • 节点的高度:该节点到其最深子节点(叶节点)的路径长度;
  • 节点的深度:根节点到该节点的路径长度

二叉搜索树

二叉搜索树的性质:左子树的节点值 < 根节点的值 < 右子树的节点值

  • 查找:找最小就一直往左走,找最大就一直往右走;
  • 插入:从根节点开始,遇到大的就向左,遇到小的就向右,一直到尾端,即为插入点。
  • 删除:分两种情况:
    • 目标节点只有一个子节点,直接将其子节点连至其父节点。
    • 目标节点有两个子节点,以右子树中的最小值取而代之即可。

平衡二叉搜索树

所谓平衡,即没有任何一个节点过深。

平衡二叉搜索树是加了“额外平衡条件”的二叉搜索树。

平衡二叉搜索树比一般的二叉搜索树要复杂,插入节点和删除节点的平均时间也更长,但是它们可以避免极难应付的最坏(高度不平衡)情况,而且由于它们总是保持着某种程度的平衡,所以元素的查找时间平均而言就比较少。

平衡二叉搜索树相较一般的二叉搜索树:

  • 查找更快;
  • 插入和删除节点较慢。

常见的平衡二叉搜索树有AVL树、红黑树。

AVL树

为了确保整棵树的深度为O(logN),AVL树的“额外平衡条件”是:

  • 任何节点的左右子树高度相差最多为1。

最佳的平衡条件肯定是每个节点的左右子树有相同高度,但这个条件过于严苛,很难保证插入新元素而又保持这种平衡。所以AVL树退而求其次,允许每个节点的左右子树高度相差1,虽然这个条件会把平衡弱化,但仍能保证“对数深度”平衡状态。

插入新节点后,可能会有节点违反平衡条件,且只有插入点到根节点这条路径上的各节点可能违反平衡状态。只要调整其中最深的那个节点,便可使整棵树重新获得平衡。

假设该最深节点为X,由于节点最多有两个子节点,而所谓“平衡被破坏”意味着X的左右子树的高度相差2,因此可以分为四种情况:

  1. 插入点位于X的左子节点的左子树——左左;
  2. 插入点位于X的左子节点的右子树——左右;
  3. 插入点位于X的右子节点的左子树——右左;
  4. 插入点位于X的右子节点的右子树——右右。

左左和右右彼此对称,称为外侧插入,可用单旋转解决;
左右和右左彼此对此,称为内侧插入,可用双旋转解决。

单旋转

对于外侧插入,以左左为例,情况肯定是这样的,X的左子节点的左子树因新节点插入而成长了一层,导致X的左子树比右子树深度多2.

调整方法(左左为例):将X的左子节点提起来,使X自然下滑,并将X的左子节点的右子树挂到下滑后的X的左侧,即可重新获得平衡。

这么做的原因呢?

  1. 二叉树的规则使我们知道,X值肯定大于X的左子节点值,所以X必须成为调整后的X的左子节点的右子节点;
  2. 同时,X的左子节点的右子树的所有节点值肯定介于X和X的左子节点的值之间,所以X的左子节点的右子树应该落在调整后的X的左侧。

可以看出,左左的单旋转是一次顺时针旋转;右右的单旋转对称,是一次逆时针旋转。

双旋转

对于内侧插入,以左右为例,单旋转无法解决问题。

调整方法:双旋转利用两次单旋转完成:

  1. 先对X的左子节点和X的左子节点的右子节点做一次单旋转;
  2. 再对X和第一次调整后的新的X的左子节点(原来的X的左子节点的右子节点)做一次单旋转。

AVL树的平衡条件仍较为严苛,对于set和map来说,需要频繁的插入和删除,若用AVL树作底层结构,会影响性能。AVL树只适合查找较多而插入删除操作较少的情况。

RB-tree(红黑树)

红黑树不仅是二叉搜索树,而且满足以下规则:

  1. 每个节点不是红色就是黑色;
  2. 根节点是黑色;
  3. 如果节点为红,其子节点必须为黑;(即父子节点不得同时为红);
  4. 任一节点至NULL的任何路径,所含黑节点树必须相同。
  • 根据规则4,新增节点必须为红;
  • 根据规则3,新增节点的父节点必须为黑。

当新节点按照二叉搜索树的规则到达插入点,却未能符合上述条件时,就必须调整颜色和旋转树形。

新插入节点导致破坏红黑树的规则肯定是破坏3,即父子节点同时为红了。
NULL节点可视为黑色。

设新节点为X,其父节点为P,伯父节点为S,祖父节点为G,曾祖父节点为GG。根据规则4,X必为红,若P亦为红(这就违反了规则3,需要调整),则G必为黑(因为原为RB-tree,得符合规则3)。那么,根据X的插入位置及外围节点(S和GG)的颜色,可分为以下四种情况:

  1. S为黑且X为外侧插入。对这种情况,只需对P、G做一次单旋转,并更改P、G颜色,即可重新满足红黑树的规则3;
  2. S为黑且X为内侧插入。对这种情况,必须先对P、X做一次单旋转,并更改G、X颜色,再将结果对G做一次单旋转,即可重新满足规则3;
  3. S为红且X为外侧插入。对这种情况,先对P、G做一次单旋转,并改变X的颜色。此时如果GG为黑,一切搞定。若GG为红,见情况4;
  4. S为红且X为外侧插入。对这种情况,先对P、G做一次单旋转,并改变X的颜色。如果此时GG为红,还得持续往上做,直到不再有父子连续为红的情况。

以红黑树为底层结构的关联容器

set

set的特性是,所有的元素都会根据key值自动被排序。

set的key值就是value值,value值就是key值。set不允许两个元素有相同的key值。

不能通过set的迭代器更改set的元素值。因为set元素值就是其key值,关系到set元素的排列规则。如果任意改变set元素值,会严重破坏set组织。set的迭代器其实被定义成底层红黑树的const迭代器了,杜绝写入操作。

对set进行插入和删除操作后,之前的迭代器不会失效。(当然,被删除元素的迭代器肯定是失效了)

map

map的特性是,所有元素会根据各自的key值自动被排序。

map的元素是pair类型,同时拥有key值和value值。pair的第一元素被视为key,第二元素被视为value。map不允许两个元素有相同的key值。

通过map的迭代器可以修改map元素的value值,不可以修改map的key值,因为key值关系到map的排列规则,任意更改map元素的key值会破坏map组织。

对map进行插入和删除操作后,之前的迭代器不会失效。(当然,被删除元素的迭代器肯定是失效了)

multiset

multiset的特性及用法与set完全相同,唯一的差别在于它允许key值重复。

multimap

multimap的特性及用法与set完全相同,唯一的差别在于它允许key值重复。

哈希表(Hash-table)

二叉搜索树具有对数平均时间的表现建立在“输入的数据有足够的随机性”这样一种假设上。

哈希表在查找、插入、删除等操作上具有“常数平均时间”的表现,且这种表现是以统计为基础的,不需依赖输入元素的随机性。

哈希函数:将哈希表中元素的键值映射为元素存储位置的函数。
A d d r e s s = H a s h ( k e y ) Address = Hash(key) Address=Hash(key)

哈希冲突:可能出现不同的元素被映射到相同的位置。

比如采用求余的哈希函数的话,2和5对3求余都是2,这样2和5就都存到位置2上了,这就是哈希冲突。

解决哈希冲突的方法:

  • 线性探测
  • 二次探测
  • 开链法

以哈希表为底层结构的关联容器

红黑树有自动排序功能,哈希表没有。所以以哈希表为底层的关联容器在C++标准里都以unordered_开头。

unordered_set

与set一样,unordered_set的key值就是value值,value值就是key值。

unordered_map

与map一样,unordered_map的元素是pair类型,同时拥有key值和value值。pair的第一元素被视为key,第二元素被视为value。

unordered_multiset

unordered_multiset与multiset的特性完全相同,唯一的差别在于其底层是哈希表,所以其元素不会被自动排序。

unordered_multimap

unordered_multimap与multimap的特性完全相同,唯一的差别在于其底层是哈希表,所以其元素不会被自动排序。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

极客熊猫GeekPanda

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值