首页 > 其他分享 >STL

STL

时间:2023-06-15 18:12:58浏览次数:23  
标签:容器 vector 迭代 deque STL 元素 num

STL概述

什么是C++标准模板库(STL)

标准模板库STL(Standard Template Library),是C++标准库的一部分,不需要单独安装,只需要#include头文件。

C++对模板(Template)支持得很好,STL就是借助模板把常用的数据结构及其算法都实现了一遍,并且做到了数据结构和算法的分离。

C++语言的核心优势之一就是便于软件的复用。

C++语言有两个方面体现了复用:

  • 面向对象的继承和多态机制

  • 通过模板的概念实现了对泛型程序设计的支持

C++中的模板,就好比英语作文的模板,只换主题,不换句式和结构。对应到C++模板,就是只换类型,不换方法。

STL的优势

STL封装了很多实用的容器,省时省力,能够让你将更多心思放到解决问题的步骤上,而非费力去实现数据结构诸多细节上

STL六大组件

1、容器(数据结构):用来存数据的。

  • 序列式容器 vector、list、deque
  • 关联式容器 set、map
  • 无序关联式容器 unordered_set、unordered_map

2、迭代器:看成是一种指针,称为泛型指针。

3、算法:可以对容器中的数据进行相应的操作。

4、适配器

  • 容器适配器 优先级队列
  • 算法适配器
  • 迭代器的适配器 (源码)

5、函数对象(仿函数):做定制化操作。

6、空间配置器(allocator):空间的申请与释放。(研究基本使用 + 原理 + 源码)

程序 = 容器(数据结构) + 算法

序列式容器

1、初始化

三种序列化容器:vector、list、deque都支持以下5种初始化的方式

// 1、创建空对象
vector<类型名> 变量名;

// 2、创建count个value
vector<int> number(10,3);	// 创建10个3

// 3、迭代器范围
int arr[5] = {1,2,3,4,5};
vector<int> number(arr,arr + 5);	// 左闭右开 [,)

// 4、拷贝构造函数与移动构造函数

// 5、大括号的方式
vector<int> number = {1,23,4,13};

2、遍历

三种序列式容器,其中:list是没有下标的,不支持下标访问运算符。其他的三种遍历的方式,三种容器都支持

vector<int> num = {1,2,3,4,5};
// 1.使用下标访问运算符进行遍历
for(size_t idx = 0;idx != num.size();++idx)
{
    cout << num[idx] << "   ";
}
cout << endl;

// 2.使用迭代器进行遍历
// vector的迭代器不支持it < num.end()的写法,因此循环条件只能it!=num.end()   
vector<int>::iterator it;   // 泛型指针
// vector<int>::iterator it = num.begin();
// for(;it != num.end();++it)
for(it = num.begin();it != num.end();++it)
{
    cout << *it << "    ";
}
cout << endl;

// 3.C++11新特性,for与auto
for(auto& elem : num)
{
    cout << elem << "   ";
}

注意:list不支持下标(因为list底层用的链表,链表是不支持随机访问的!)

3、插入与删除

3.1、在尾部插入和删除

三种容器在尾部进行插入与删除的操作是完全一样的

vector<int> num = {1,2,3,4,5};
num.push_back(6);	// 插入
num.pop_back();		// 删除

3.2、在头部插入和删除

deque与list支持在头部进行插入与删除,但是vector没有在头部进行插入与删除的操作

list<int> num = {1,2,3,4,5};
num.push_front(0);
num.pop_front();

Q:为什么vector不支持在头部进行插入和删除操作?

A:因为vector中的元素是连续的,如果在头部插入(删除)元素,那么后面的所有元素都要后移(前移),时间复杂度是O(N)

3.3、在任意位置插入

三种序列式容器在任意位置插入元素的四种方法

语法格式 用法说明
iterator insert(pos,elem) 在迭代器 pos 指定的位置之前插入一个新元素elem,并返回表示新插入元素位置的迭代器。
iterator insert(pos,n,elem) 在迭代器 pos 指定的位置之前插入 n 个元素 elem,并返回表示第一个新插入元素位置的迭代器。
iterator insert(pos,first,last) 在迭代器 pos 指定的位置之前,插入其他容器(不仅限于vector)中位于 [first,last) 区域的所有元素,并返回表示第一个新插入元素位置的迭代器。
iterator insert(pos,initlist) 在迭代器 pos 指定的位置之前,插入初始化列表initlist(用大括号{}括起来的多个元素,中间有逗号隔开)中所有的元素,并返回表示第一个新插入元素位置的迭代器。
std::vector<int> demo{1,2};
//第一种格式用法
demo.insert(demo.begin() + 1, 3);//{1,3,2}
//第二种格式用法
demo.insert(demo.end(), 2, 5);//{1,3,2,5,5}
//第三种格式用法
std::array<int,3> test{ 7,8,9 };
demo.insert(demo.end(), test.begin(), test.end());//{1,3,2,5,5,7,8,9}
//第四种格式用法
demo.insert(demo.end(), { 10,11 });//{1,3,2,5,5,7,8,9,10,11}

emplace()

emplace() 是C++11 标准新增加的成员函数,用于在 vector 容器指定位置之前插入一个新的元素。

再次强调,emplace() 每次只能插入一个元素,而不是多个。

该函数的语法格式如下:

iterator emplace (const_iterator pos, args...);

其中,pos 为指定插入位置的迭代器;args... 表示与新插入元素的构造函数相对应的多个参数;该函数会返回表示新插入元素位置的迭代器。

简单的理解 args...,即被插入元素的构造函数需要多少个参数,那么在 emplace() 的第一个参数的后面,就需要传入相应数量的参数。

3.4、vector容器在任意位置删除元素

// 删除 vector 容器中 pos 迭代器指定位置处的元素,并返回指向被删除元素下一个位置元素的迭代器。该容器的大小(size)会减 1,但容量(capacity)不会发生改变。
iterator erase(pos);


// 用erase函数删除vector中连续重复的元素
void test()
{
    vector<int> num = {1,3,6,6,6,6,6,6,7,5};
    // 注意这里不能 ++it
    for(auto it = num.begain();it != num.end();)
    {
        if(6 == *it)
        {
            num.erase(it);
        }
        else
        {
            ++it;
        }
    }
}

4、元素的清空

// 删除容器中所有的元素,使其变成空的容器。(没有回收内存)
void clear();
// 回收空间
void shrink_to_fit();

对于deque和vector,使用clear函数只是将元素清空,但并没有回收内存,需要配合shrink_to_fit()函数将内存回收;

对于list,元素删除了结点就会删除,不存在回收空间的函数。

5、vector的原理剖析

  • _M_start,指向第一个元素;

  • _M_finish,指向最后一个元素的下一个位置;

  • _M_end_of_storage,最后一个空间的下一个位置

template <class _Tp, class _Alloc = __STL_DEFAULT_ALLOCATOR(_Tp) >
class vector
{
public:
    //类型萃取(重要)
    typedef _Tp value_type;
    typedef value_type* pointer;
    typedef const value_type* const_pointer;
    typedef value_type* iterator;
    typedef const value_type* const_iterator;
    typedef value_type& reference;
    typedef const value_type& const_reference;
    typedef size_t size_type;
    typedef ptrdiff_t difference_type;
    
    //为了严格表明_Base::allocator_type是一种类型,所以加上typename关键字进行强调
    typedef typename _Base::allocator_type allocator_type;
    
    //vector的下标访问运算符的重载
    reference operator[](size_type __n)
    { 
        return *(begin() + __n); 
    }
    
    void _M_range_check(size_type __n) const 
    {
       if (__n >= this->size())
           __stl_throw_range_error("vector");
    }

    //总结:通过随机获取元素,可以使用at,也可以直接使用下标,但是at比下标访问更安全,因为有范围检查
    reference at(size_type __n)
    {
		_M_range_check(__n); 
		return (*this)[__n]; 
    }
};

6、deque的原理剖析

和 vector 容器采用连续的线性空间不同,deque 容器存储数据的空间是由一段一段等长的连续空间构成,各段空间之间并不一定是连续的,可以位于在内存的不同区域(片段内部是连续的,片段之间是不连续,即逻辑上是连续的,但是物理上是不连续)。

为了管理这些连续空间,deque 容器用数组(数组名假设为 map)存储着各个连续空间的首地址。也就是说,map 数组(中控器)中存储的都是指针,指向那些真正用来存储数据的各个连续空间

通过建立 map 数组,deque 容器申请的这些分段的连续空间就能实现“整体连续”的效果。换句话说,当 deque 容器需要在头部或尾部增加存储空间时,它会申请一段新的连续空间,同时在 map 数组的开头或结尾添加指向该空间的指针,由此该空间就串接到了 deque 容器的头部或尾部

deque容器迭代器的底层实现

由于 deque 容器底层将序列中的元素分别存储到了不同段的连续空间中,因此要想实现迭代器的功能,必须先解决如下 2 个问题:

  1. 迭代器在遍历 deque 容器时,必须能够确认各个连续空间在 map 数组中的位置;
  2. 迭代器在遍历某个具体的连续空间时,必须能够判断自己是否已经处于空间的边缘位置。如果是,则一旦前进或者后退,就需要跳跃到上一个或者下一个连续空间中。

为了实现遍历 deque 容器的功能,deque 迭代器定义了如下的结构:

template <class _Tp, class _Ref, class _Ptr>
struct _Deque_iterator {
  /*......*/
  _Tp* _M_cur;
  _Tp* _M_first;
  _Tp* _M_last;
  _Map_pointer _M_node;		//_Map_pointer 等价于 T**
  /*......*/
}

迭代器内部包含 4 个指针,它们各自的作用为:

  • cur:指向当前正在遍历的元素;
  • first:指向当前连续空间的首地址;
  • last:指向当前连续空间的末尾地址;
  • node:它是一个二级指针,用于指向 map 数组中存储的指向当前连续空间的指针。

借助这 4 个指针,deque 迭代器对随机访问迭代器支持的各种运算符进行了重载,能够对 deque 分段连续空间中存储的元素进行遍历。

deque容器的底层实现

deque 容器除了维护 map 数组,还需要维护 start、finish 这 2 个 deque 迭代器。以下为 deque 容器的定义:

template <class _Tp, class _Alloc = __STL_DEFAULT_ALLOCATOR(_Tp) >
class deque : protected _Deque_base<_Tp, _Alloc> {
// ...
protected:
  _Tp** _M_map;
  iterator _M_start;
  iterator _M_finish;
}
  • start 迭代器记录着 map 数组中首个连续空间的信息
  • finish 迭代器记录着 map 数组中最后一个连续空间的信息

另外需要注意的是,和普通 deque 迭代器不同,start 迭代器中的 cur 指针指向的是连续空间中首个元素;而 finish 迭代器中的 cur 指针指向的是连续空间最后一个元素的下一个位置

借助 start 和 finish,以及 deque 迭代器中重载的诸多运算符,就可以实现 deque 容器提供的大部分成员函数。

7、迭代器失效问题

迭代器的失效问题:对容器的操作影响了元素的存放位置,称为迭代器失效。

失效情况:

  1. 当容器调用erase()方法后,当前位置到容器末尾元素的所有迭代器全部失效。
  2. 当容器调用insert()方法后,当前位置到容器末尾元素的所有迭代器全部失效。
  3. 如果容器扩容,在其他地方重新又开辟了一块内存。原来容器底层的内存上所保存的迭代器全都失效了。
  4. 不同容器的迭代器,是不能进行比较运算的。

对于list而言,插入的时候,迭代器是跟着结点走了,它在使用的时候,不会存在什么问题。但是对于vector而言,插入元素的大小会影响vector底层的扩容,有可能会导致迭代器失效。所以为了解决这个问题,每次操作迭代器时候,最好可以提前更新迭代器的位置,让迭代器指向新的空间的位置,这样就保证不会出错。

特别的,对于deque而言,去进行insert操作的时候,也有可能迭代器失效,所以最好还是可以每次使用迭代器的时候,像vector一样,将迭代器的位置进行更新,指向新的空间的位置。

8、list的特殊用法

8.1、sort函数

void sort();

template< class Compare >
void sort( Compare comp );

template<typename T>
struct CompareList
{
    // 参数中的const表明该函数不能修改lhs和rhs
    // 第二个const会将成员函数声明为常量成员函数,表明这个函数内不允许修改类的数据成员
    bool operator()(const T& lhs,const T& rhs) const
    {
        return lhs < rhs;
    }
};

template<typename T>
void display(T& num)
{
    for(auto& elem : num)
    {
        cout << elem << "   ";
    }
    cout << endl;
}

void test()
{
    list<int> num = {9,37,19,0,1,7,8,7,2,3,4};
    display(num);
    num.sort(CompareList<int>());   // 自己实现比较规则
    display(num);
}
  • num.sort(),默认情况,会按照从小到大的顺序进行排序
  • num.sort(std::less<int>()),按照小于的比较规则进行比较(从小到大排序)
  • num.sort(std::greate<int>()),按照大于的比较规则进行比较(从大到小排序)
  • 自己实现比较规则

8.2、merge函数

// 两个链表进行merge的时候,需要分别将两个链表从小到大排序,然后再进行merge,以防达不到合并后排序的效果
void merge( list& other );

void test()
{
    list<int> num = {9,37,19,0,1,7,8,7,2,3,4};
    list<int> num1 = {9,2,34,5,323,3234,54523,1342,32};
    num.sort();
    num1.sort();
    num.merge(num1);
    display(num);
}

8.3、splice函数

list::splice实现list拼接的功能。将源list的内容部分或全部元素删除,并插入到目的list。

 int main()
 {
	 list<int>li1{ 1,2,3,4 };
	 list<int>li2{ 11,12,13,14 };
 
	 list<int>::iterator it = li1.begin();
	 it++; // 插入到第一个元素后面
 
	 li1.splice(it, li2);
	 if (li2.empty()) {
		 cout << "li2 is empty" << endl;
	 }
	 else
	 {
		 cout << "li2 is  not empty" << endl;
	 }
 
	 cout << "li1 is" << endl;
	 for (list<int>::iterator item = li1.begin(); item != li1.end(); ++item) {
		 cout << *item << " ";
	 }
     return 0;
}

8.4、其他

void reverse() noexcept;		// 反转容器中元素的顺序。没有引用或迭代器失效。
void unique();					// 去除重复元素(需要先将元素排序)

标签:容器,vector,迭代,deque,STL,元素,num
From: https://www.cnblogs.com/MyXjil/p/17483697.html

相关文章

  • 【C++】STL常用容器总结之十二:string类
    13、string类声明string类本不是STL的容器,但是它与STL容器有着很多相似的操作,因此,把string放在这里一起进行介绍。之所以抛弃char*的字符串而选用C++标准程序库中的string类,是因为他和前者比较起来,不必担心内存是否足够、字符串长度等等,而且作为一个类出现,他集成的操作函数足以完......
  • STL-algorithm(ACM)
    unique(a.begin(),a.end())待研究与离散化有关//翻转(reverse(位置,位置))reverse(a.begin(),a.end());inta[5]={1,2,3,4,5};reverse(a,a+5);//结果54321循环移位(rotate(移动到该位置前一个地方,移动区间的头位置,移动的尾位置))第一个位置必须在第二......
  • BouncyCastle
    在openEuler(推荐)或Ubuntu或Windows(不推荐)中完成下面任务参考附件内容完成SM2加解密的内容,提交运行结果截图(10‘)2完成SM3,SM4算法的调用,提交运行结果截图和代码(15’,选做)BouncyCastle配置1.jar包下载官网:https://www.bouncycastle.org/latest_releases.htmlbcprov-e......
  • STL-string(ACM)
    1.相当于加了一些操作的vector<char>基本操作字符串转换(C++11)//将字符串转换为整型stoi()//将字符串转换为longlongstoll()//将字符串转换为float型stof()//将字符串转换为double型stod()后面加入s+=t;//时间复杂度O(t)s.push_back();字符串替换......
  • STL-multiset(ACM)
    1.与set不同的是,multiset可以允许多个相同元素同时出现重载函数(默认)multiset<int,int>mu;基本操作mu.erase(x);//把所有与x相同的元素删除//如果我们只想删除一个的话//通过删除迭代器实现mu.erase(mu.find(x));mu.count(x);//查的时间与x的个数有关,慎用需......
  • BouncyCastle
    任务详情在openEuler(推荐)或Ubuntu或Windows(不推荐)中完成下面任务参考附件内容完成SM2加解密的内容,提交运行结果截图(10‘)完成SM3,SM4算法的调用,提交运行结果截图和代码(15’,选做)BouncyCastle配置1.jar包下载官网:https://www.bouncycastle.org/latest_releases.html......
  • STL-map(ACM)
    1.不存在的元素查询时会自动生成2.map就是一堆pair的集合,按照T1的字典序进行排列3.可以像vector那样根据下标随时访问重载函数 map<T1,T2>m;//下标的类型,值的类型//按照T1的值进行字典序排序//下方为赋值操作map<string,string>m;m["AC"]="Yee";m["Red"]=......
  • STL之Stack与queue的模拟实现与duque的底层结构(3千字长文详解)
    STL之Stack与queue的模拟实现与duque的底层结构设计模式的概念设计模式像是古代的兵法,是以前的人总结出来的一些在特定的情况下,某种特定的好用的方法总结STL中迭代器也是一种设计模式——==迭代器模式==STL中stack和queue的实现就是使用了一种设计模式——==适配器模式!==适......
  • STL之优先级队列(堆)的模拟实现与仿函数(8千字长文详解!)
    STL之优先级队列(堆)的模拟实现与仿函数优先级队列的概念优先队列是一种==容器适配器==,根据严格的弱排序标准,==它的第一个元素总是它所包含的元素中最大的==。本质就是一个堆!此上下文类似于堆,在堆中可以随时插入元素,并且只能检索最大堆元素(优先队列中位于顶部的元素)。......
  • STL之反向迭代器的实现
    反向迭代器的实现反向迭代器介绍反向迭代器和正向迭代器的区别就是反向迭代器的++/--是倒着走!那么反向迭代器的本质是什么?——==是一个容器适配器!==本质就是通过==封装迭代器==,来让其实现我们想要的行为!也可以通过通过复制一份正向迭代器,然后修改正向迭代器的行为,来实现反......