C++初阶学习第十一弹——探索STL奥秘(六)——深度刨析list的用法和核心点

简介: C++初阶学习第十一弹——探索STL奥秘(六)——深度刨析list的用法和核心点

前言:

在前面,我们已经学习了STL中的string和vector,现在就来讲解STL中的最后一个部分——list的使用及其相关知识点,先说明一点,因为我们之前已经讲过了string和vector的接口函数等用法,list的这些用法与它们相差不大,所以我们讲解的重心就不再是如何使用list上,而是后面list的模拟实现和一些细节点


一、list的使用

1.1 list的简单接口函数

首先我们需要先明确list的底部实际上是类似一个带头双向链表的,结构如下图所示:

因此list非常便于插入和删除数据,下面我们就先来看一下list的一些重要的接口函数

初始化列表:

std::list<int> myList = {1, 2, 3, 4, 5};

通过迭代器访问元素:

std::list<int>::iterator it = myList.begin();
while (it != myList.end()) {
    std::cout << *it << std::endl;
    ++it;
}

在链表尾部插入元素:

myList.push_back(6);

在链表头部插入元素:

myList.push_front(0);

删除元素:

myList.remove(3); // 删除值为3的元素
myList.erase(it); // 删除迭代器指向的元素

排序链表:

myList.sort();

反转链表:

myList.reverse();

上面这些就是list经常使用的一些接口函数,没啥难度,有不理解的地方可以私信我或者到网上搜一下

1.2 list的注意事项

  • 迭代器失效: 在list进行插入和删除操作时,不仅操作的元素所在的迭代器会失效,所有指向链表的迭代器、指针和引用都会失效。因此,在进行操作后,需要重新获取有效的迭代器。(vector的使用也要注意这个问题)
  • 内存效率: list的内存效率相对较高,因为它不需要像数组那样连续分配内存,但是它的插入和删除操作的时间复杂度为O(1),这是因为链表的每个元素都需要存储指向前后节点的指针。
  • 没有容量概念: list没有容量(capacity)这个概念,它总是根据需要动态分配内存。
  • 元素唯一性: list中的元素是不重复的,如果尝试插入已经存在的元素,该元素将被覆盖。
  • 操作顺序: 由于list是双向链表,因此插入和删除操作会保持元素的相对顺序,即元素在链表中的位置不会改变。

使用list时,应该根据具体需求选择合适的操作,并注意迭代器的管理,以确保程序的正确性。

特别强调一下迭代器失效的问题,list的迭代器失效问题一般只有在删除元素的时候会出现,因为它插入数据的时候都是开辟的新空间,不同数据之间一般不是连接在一起的


二、list的模拟实现

list的模拟实现上与前面的vector和string也极为相似,这里我们主要想讲一下list的迭代器的模拟实现,首先我们要知道,因为我们期待迭代器能像指针那样发挥作用,所以它的模拟实现需要包含以下几点:


1. 指针可以解引用,迭代器的类中必须重载operator*()


2. 指针可以通过->访问其所指空间成员,迭代器类中必须重载oprator->()


3. 指针可以++向后移动,迭代器类中必须重载operator++()与operator++(int)


至于operator--()/operator--(int)释放需要重载,根据具体的结构来抉择,双向链表可以向前 移动,所以需要重载,如果是forward_list就不需要重载--


4. 迭代器需要进行是否相等的比较,因此还需要重载operator==()与operator!=()

list迭代器也要分为两种:正向迭代器和反向迭代器

因为list正反向迭代器的应用都要实现,所以还是比较麻烦的,下面我们直接来看一下实现

#include <iostream>
using namespace std;
#include <assert.h>
namespace zda
{
  // List的节点类
  template<class T>
  struct ListNode
  {
    ListNode(const T& val = T())
      : _prev(nullptr)
      , _next(nullptr)
      , _val(val)
    {}
 
    ListNode<T>* _prev;
    ListNode<T>* _next;
    T _val;
  };
 
  //正向迭代器
  template<class T, class Ref, class Ptr>
  class ListIterator
  {
    typedef ListNode<T> Node;
    typedef ListIterator<T, Ref, Ptr> Self;
 
    // Ref 和 Ptr 类型需要重定义下,实现反向迭代器时需要用到
  public:
    typedef Ref Ref;
    typedef Ptr Ptr;
  public:
    //
    // 构造
    ListIterator(Node* node = nullptr)
      : _node(node)
    {}
 
    //
    // 具有指针类似行为
    Ref operator*()
    {
      return _node->_val;
    }
 
    Ptr operator->()
    {
      return &(operator*());
    }
 
    // 迭代器支持移动
    Self& operator++()
    {
      _node = _node->_next;
      return *this;
    }
 
    Self operator++(int)
    {
      Self temp(*this);
      _node = _node->_next;
      return temp;
    }
 
    Self& operator--()
    {
      _node = _node->_prev;
      return *this;
    }
 
    Self operator--(int)
    {
      Self temp(*this);
      _node = _node->_prev;
      return temp;
    }
 
    // 迭代器支持比较
    bool operator!=(const Self& l)const
    {
      return _node != l._node;
    }
 
    bool operator==(const Self& l)const
    {
      return _node != l._node;
    }
 
    Node* _node;
  };
 
  //反向迭代器
  template<class Iterator>
  class ReverseListIterator
  {
  public:
    typedef typename Iterator::Ref Ref;
    typedef typename Iterator::Ptr Ptr;
    typedef ReverseListIterator<Iterator> Self;
  public:
    // 构造
    ReverseListIterator(Iterator it)
      : _it(it)
    {}
 
    // 具有指针类似行为
    Ref operator*()
    {
      Iterator temp(_it);
      --temp;
      return *temp;
    }
 
    Ptr operator->()
    {
      return &(operator*());
    }
 
    // 迭代器支持移动
    Self& operator++()
    {
      --_it;
      return *this;
    }
 
    Self operator++(int)
    {
      Self temp(*this);
      --_it;
      return temp;
    }
 
    Self& operator--()
    {
      ++_it;
      return *this;
    }
 
    Self operator--(int)
    {
      Self temp(*this);
      ++_it;
      return temp;
    }
 
    // 迭代器支持比较
    bool operator!=(const Self& l)const
    {
      return _it != l._it;
    }
 
    bool operator==(const Self& l)const
    {
      return _it != l._it;
    }
 
    Iterator _it;
  };
}


三、list和vector的区别

1、任意位置插入删除时:list可以随意插入删除,但是vector任意位置的插入删除效率低,需要挪动元素,尤其是插入时有时候需要异地扩容,就需要开辟新空间,拷贝元素,释放旧空间,效率很低

2、访问元素时:vector支持随机访问,但是list不支持随机访问

3、迭代器的使用上:vector可以使用原生指针,但是list需要对原生指针进行封装

4、空间利用上:vector使用的是一个连续的空间,空间利用率高,而list使用的是零碎的空间,空间利用率低

四、总结

以上就是学习list的一些重点内容和基本操作,这些内容当然是不全的,剩下有很多内容需要自己再去学习一下,后期我也会有针对的再加一些内容进来

感谢大佬观看,创作不易,还请各位大佬一键三连!!!

相关文章
|
6月前
|
存储 Java 索引
(Python基础)新时代语言!一起学习Python吧!(二):字符编码由来;Python字符串、字符串格式化;list集合和tuple元组区别
字符编码 我们要清楚,计算机最开始的表达都是由二进制而来 我们要想通过二进制来表示我们熟知的字符看看以下的变化 例如: 1 的二进制编码为 0000 0001 我们通过A这个字符,让其在计算机内部存储(现如今,A 字符在地址通常表示为65) 现在拿A举例: 在计算机内部 A字符,它本身表示为 65这个数,在计算机底层会转为二进制码 也意味着A字符在底层表示为 1000001 通过这样的字符表示进行转换,逐步发展为拥有127个字符的编码存储到计算机中,这个编码表也被称为ASCII编码。 但随时代变迁,ASCII编码逐渐暴露短板,全球有上百种语言,光是ASCII编码并不能够满足需求
288 4
|
算法 C++ 容器
模拟实现c++中的list模版
模拟实现c++中的list模版
|
算法 网络安全 区块链
2023/11/10学习记录-C/C++对称分组加密DES
本文介绍了对称分组加密的常见算法(如DES、3DES、AES和国密SM4)及其应用场景,包括文件和视频加密、比特币私钥加密、消息和配置项加密及SSL通信加密。文章还详细展示了如何使用异或实现一个简易的对称加密算法,并通过示例代码演示了DES算法在ECB和CBC模式下的加密和解密过程,以及如何封装DES实现CBC和ECB的PKCS7Padding分块填充。
389 4
2023/11/10学习记录-C/C++对称分组加密DES
|
C++ 开发者
C++学习之继承
通过继承,C++可以实现代码重用、扩展类的功能并支持多态性。理解继承的类型、重写与重载、多重继承及其相关问题,对于掌握C++面向对象编程至关重要。希望本文能为您的C++学习和开发提供实用的指导。
225 16
|
编译器 C语言 C++
【c++丨STL】list模拟实现(附源码)
本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
341 2
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
472 7
|
存储 编译器 C++
C++ initializer_list&&类型推导
在 C++ 中,`initializer_list` 提供了一种方便的方式来初始化容器和传递参数,而右值引用则是实现高效资源管理和移动语义的关键特性。尽管在实际应用中 `initializer_list&&` 并不常见,但理解其类型推导和使用方式有助于深入掌握现代 C++ 的高级特性。
232 4
|
编译器 C语言 C++
配置C++的学习环境
【10月更文挑战第18天】如果想要学习C++语言,那就需要配置必要的环境和相关的软件,才可以帮助自己更好的掌握语法知识。 一、本地环境设置 如果您想要设置 C++ 语言环境,您需要确保电脑上有以下两款可用的软件,文本编辑器和 C++ 编译器。 二、文本编辑器 通过编辑器创建的文件通常称为源文件,源文件包含程序源代码。 C++ 程序的源文件通常使用扩展名 .cpp、.cp 或 .c。 在开始编程之前,请确保您有一个文本编辑器,且有足够的经验来编写一个计算机程序,然后把它保存在一个文件中,编译并执行它。 Visual Studio Code:虽然它是一个通用的文本编辑器,但它有很多插
599 6
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
472 12
下一篇
开通oss服务