STL容器之list

news2024/11/20 12:18:30

​ 1.封装除了对数据的保护、更好地管理数据之外,还有实现了对上层的统一;

​ 2.类模板参数的不同,一方面是为了实例化出来不同的类,另一方面是为了实现类的成员函数的不同;

一、认识list

​ 1.list是一种带头双向循环链表。相较于vector的最大优点就是支持了头插头删,而且时间复杂度是O1;2.list不支持+,因为链表不是一段连续的空间,重载实现得遍历一遍,代价太大,但是却支持了迭代器++,可以用迭代器遍历查找,或者算法find配合查找;3.迭代器循环访问使用!=不是<,是因为空间可能是不连续的;4.insert不存在迭代器失效问题,erase存在。

1.默认成员函数

explicit list (const allocator_type& alloc = allocator_type());
explicit list (size_type n, const value_type& val = value_type(),const allocator_type& alloc = allocator_type());
template <class InputIterator>
list (InputIterator first, InputIterator last,const allocator_type& alloc = allocator_type());
list (const list& x);

2.迭代器

iterator begin();
const_iterator begin() const;
iterator end();
const_iterator end() const;
reverse_iterator rbegin();
const_reverse_iterator rbegin() const;
reverse_iterator rend();
const_reverse_iterator rend() const;

3.空间

bool empty() const;
size_type size() const;

4.元素访问

reference front();
const_reference front() const;
reference back();
const_reference back() const;

5.成员类型

​ InputIterator涉及到父子类关系,一般都可以使用。

​ 迭代器按容器底层结构分为:单向迭代器可以++,双向迭代器可以++ --,随机迭代器可以++ – + -。

​ 单向迭代器forward iterator:forward_list/(哈希)unordered_map/unordered_set

​ 双向迭代器bidirectional iterator:list/map/set

​ 随机迭代器random iterator:vector/string/dequeue/

//与vector类似,除了迭代器类型发生了变化,变成了双向迭代器

在这里插入图片描述

6.成员修改

void push_front (const value_type& val);
void pop_front();
void push_back (const value_type& val);
void pop_back();
iterator insert (iterator position, const value_type& val);
void insert (iterator position, size_type n, const value_type& val);//配合算法的find(迭代器区间),迭代器区间的设计统一为左闭右开。
template <class InputIterator>
void insert (iterator position, InputIterator first, InputIterator last);
iterator erase (iterator position);
iterator erase (iterator first, iterator last);

7.其他操作

reserve()//与库里面实现的功能是一样的,较为冗余
sort()//算法库里面的不支持使用的是快排,因为库里面使用的是随机迭代器,而list的迭代器是双向的迭代器不支持。list的sort底层使用的是归并排序
//如果要排序,最好用vector,效率远大于list。
merge()//归并,使用前需要先排序,归并结果才正确
unique()//去重,使用前需要先排序,效率会提高
remove()//find+erase,找到了删除,找不到什么都不做
splice()//将一个链表的节点,转移到另一个链表,全部转移,转移一个节点,区间转移

二、模拟实现list

1.命名空间

namespace List
{
    template <class T>
    struct list_node
    {
        list_node(const T &val = T()) : next_(nullptr), prev_(nullptr), val_(val) {}
        list_node<T> *next_;
        list_node<T> *prev_;
        T val_;
    };

    template <class T>
    struct list_iterator
    {
        //迭代器内嵌类型
        typedef list_node<T> node;
		// 迭代器的构造函数
        __list_iterator(node *node) : node_(node) {}
        // 迭代器的成员
        node *node;
    };
    
    template <class T>
    class list
    {
        typedef list_node<T> node;
        
	public:
        typedef __list_iterator<T, T &> iterator;
        typedef __list_iterator<T, const T &> const_iterator;
        
    public:
        list() : head_(nullptr)
        {
            head_ = new node;
            head_->next_ = head_;
            head_->prev_ = head_;
            head_->val_ = T();
        }
		~list()
        {
            delete head;
            head
        }
        
    private:
        node *head_;
    };
}

2.成员变量

private:
	node *head_;
	size_t size_;

3.普通成员函数

void push_back(const T &val)
{
    insert(end(), val);
}
void push_front(const T &val)
{
    insert(begin(), val);
}
void pop_back()
{
    erase(--end());
}
void pop_front()
{
    erase(begin());
}
iterator insert(iterator pos, const T &val)
{
    // 对于带头双向循环链表不需要检查pos的合法性。
    // 1.将迭代器转换为节点指针,便于访问数据
    node *cur = pos.node_;
    // 2.创建新节点
    node *newnode = new node(val);
    // 3.建立连接
    node *prev = cur->prev_;
    prev->next_ = newnode;
    newnode->prev_ = prev;
    newnode->next_ = cur;
    cur->prev_ = newnode;
    size_++;
    return newnode;
}
iterator erase(iterator pos)
{
    assert(pos != end()); // 不能删除哨兵位头节点
    node *cur = pos.node_;
    node *prev = cur->prev_;
    node *next = cur->next_;
    prev->next_ = next;
    next->prev_ = prev;
    delete cur;
    cur = nullptr;
    size_--;
    return next;
}
size_t size()
{
    // 每次都要遍历,时间复杂度是On
    //  size_t sz = 0;
    //  iterator it = begin();
    //  while (it != end())
    //  {
    //      ++sz;
    //      ++it;
    //  }
    //  return sz;
    return size_;
}
void clear()
{
    iterator it = begin();
    while (it != end())
    {
        it = erase(it);
    }
}
void empty_init()
{
    head_ = new node;
    head_->next_ = head_;
    head_->prev_ = head_;
}
void swap(list<T> &lt)
{
    std::swap(head_, lt.head_);
    std::swap(size_, lt.size_);
}

4.迭代器类的定义

//1.不写拷贝构造,内置类型浅拷贝已经满足,析构函数对内置类型不做处理,这样使得迭代器类满足只访问和修改,不参与节点的创建和销毁
//2.vector与string使用指针做迭代器,是因为它们的底层结构使用指针天然支持迭代器,比如 ++、!=、*等,都是直接就满足要求的,而list需要自己设计,来符合这种上层统一的像指针一样访问数据的方法。
template <class T>
struct __list_iterator
{
   		// 迭代器的内嵌类型
        typedef list_node<T> node;
        typedef __list_iterator<T, Ref, Ptr> self;

        // 迭代器的构造函数
        __list_iterator(node *node) : node_(node) {}

        // 迭代器的普通成员函数
        Ref operator*()
        {
            return node_->val_;
        }
        // 如果value是一个自定义对象,而想要访问的是对象的成员,但是直接解引用就是对象本身,而不是对象的成员,还需要对象.成员的操作来访问
        // 使用箭头就可以直接访问到成员
        // iterator->返回的是val的地址,iterator->成员转化成了地址成员,应该是地址->成员,所以猜测编译器进行了特殊处理,将iterator->成员识别
        // 成了iterator->->成员,即为了增强可读性,做了优化
        Ptr operator->()
        {
            return &(node_->val_);
        }
        self &operator++()
        {
            node_ = node_->next_;
            return *this;
        }
        self operator++(int)
        {
            self tmp(node_);
            node_ = node_->next_;
            return tmp;
        }
        self &operator--()
        {
            node_ = node_->prev_;
            return *this;
        }
        self operator--(int)
        {
            self tmp(node_);
            node_ = node_->prev_;
            return tmp;
        }
        bool operator!=(const self &it) const
        {
            return node_ != it.node_;
        }
        bool operator==(const self &it) const
        {
            return node_ == it.node_;
        }

        // 迭代器的成员
        node *node_;
};
//const迭代器的设计
//1.首先const迭代器本身要允许++,即允许自身改变,而typedef const __list_iterator<T> iterator;
//则是对象本身不允许修改。2.只要改变*运算符重载,返回值const修饰,不允许修改即可。3.直接设计const迭代器类,过于冗余,可以增加模板参数来实现对返回值和参数的修改。

5.迭代器

iterator begin()
{
    return head_->next_; // 隐式类型转换加拷贝构造,优化为构造
}
iterator end()
{
    return head_;
}
const_iterator begin() const
{
    return head_->next_; 
}
const_iterator end() const
{
    return head_;
}

6.默认成员函数

// 默认成员函数
list() : head_(nullptr), size_(0)
{
    empty_init();
}
// list(const list &lt) : head_(nullptr), size_(0)
list(const list<T> &lt) : head_(nullptr), size_(0)
{
    // 初始化哨兵位头节点
    empty_init();
    // 遍历拷贝
    for (auto &e : lt)
    {
        push_back(e);
        // list内部存放的是一个头节点,可以通过头节点找到并访问普通节点中所存放的值。而vector内部一大块空间,所以可以实现赋值。
    }
}
// list &operator=(list lt)//在类内,语法上支持直接用类名替换类型,成员函数也支持
list<T> &operator=(list<T> lt)
{
    swap(lt);
    return *this;
}
~list()
{
    clear();
    delete head_;
    head_ = nullptr;
}

7.内嵌类型

private:
	typedef list_node<T> node;
public:
	typedef __list_iterator<T, T &> iterator;
    typedef __list_iterator<T, const T &> const_iterator;

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

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

相关文章

软件实例,物流货运配货单打印模板软件单据打印查询管理系统软件教程,可以同时打印标签或补打

软件实例&#xff0c;物流货运配货单打印模板软件单据打印查询管理系统软件教程&#xff0c;可以同时打印标签或补打 一、前言 以下软件教程以 佳易王物流单打印查询系统V17.1为例说明 软件文件下载可以点击最下方官网卡片——软件下载——试用版软件下载 这个版本在原来基…

opencv绘制基本图形,在图片上添加文字

文章目录 1.opencv绘制基本图形1. 画直线&#xff0c; cv2.line( )2. 画长方形&#xff0c;cv2.rectangle( )3. 画圆型&#xff0c;cv2.circle( )4. 画折线&#xff0c;cv2.polylines( ) 2.图片上显示文字 本章主要阐述利用opencv绘制一些常见的图形方法和技巧&#xff0c;以及…

2024年贵州省事业单位考试下周一开始报名,千万不要错过报名时间

2024年贵州省事业单位考试公告已出&#xff01;快看看你能不能报名&#xff01; 1、报名时间安排 (一)网上报名 2024年2月26日-2024年2月28日 (二)网上资格初审 2024年2月26日-2024年2月29日 (三)网上缴费 2024年2月26日-2024年3月1日 2、笔试安排 2024年3月30日 08:30-10:…

HDL FPGA 学习 - FPGA基本要素,开发流程,Verilog语法和规范、编写技巧

目录 Altera FPGA 基本要素 FPGA 开发流程和适用范围 设计和实施规范 顶层设计的要点 Verilog HDL 语法规范 编写规范 设计技巧 编辑整理 by Staok&#xff0c;始于 2021.2 且无终稿。转载请注明作者及出处。整理不易&#xff0c;请多支持。 本文件是“瞰百易”计划的…

如何正确设置CrossOver之偏好设置 crossover软件使用 crossover设定 crossover软件安装

CrossOver的核心是Wine&#xff0c;Wine是一个能在多种POSIX-compliant操作系统&#xff08;如&#xff1a;Linux、Mac OS等&#xff09;上运行Windows应用的兼容层。Wine不是Windows的模拟工具&#xff0c;它是把Windows API 调用翻译成为动态的 POSIX 调用&#xff0c;实现Li…

东方博宜 1086. 姐妹数对

东方博宜 1086. 姐妹数对 思路&#xff1a;先按照题意把规律找出来&#xff0c;按照规律再写程序。 #include<iostream> using namespace std; int main() {int n ;cin >> n ;int sum 0 ;for(int i 1 ; i < n ; i){for(int j i1 ; j < n ; j){int m ;m …

Linux配置jdk、tomcat、mysql离线安装与启动

目录 1.jdk安装 2.tomcat的安装&#xff08;开机自启动&#xff09; 3.MySQL的安装 4.连接项目 1.jdk安装 上传jdk安装包 jdk-8u151-linux-x64.tar.gz 进入opt目录&#xff0c;将安装包拖进去 解压安装包 这里需要解压到usr/local目录下&#xff0c;在这里我新建一个文件夹…

Git的基本操作和原理

目录 写在前面的话 为什么要有Git&#xff08;git初识&#xff09;&#xff1f; Git安装(Centos为例) Git基本操作 创建Git本地仓库 Git配置 认识工作区、暂存区、版本库 概念认识 添加文件 查看.git文件 修改文件 版本回退 撤销修改 情况一&#xff1a;…

【办公类-22-06】周计划系列(1)“信息窗” (2024年调整版本)

作品展示 调用原来的信息窗素材&#xff0c;制作下学期的19周的信息窗基础word 背景需求&#xff1a; 开学了&#xff0c;继续做周计划系列&#xff0c;在原有基础上&#xff0c;进行进一步代码优化 【办公类-22-01】周计划系列&#xff08;1&#xff09;-生成“信息窗”&am…

供应链大数据:穿越经济迷雾的指南针

随着经济形势的变幻莫测&#xff0c;企业运营面临着前所未有的挑战。在这个充满不确定性的时代&#xff0c;供应链大数据如同一盏明亮的指南针&#xff0c;为企业提供精准的方向指引。下面&#xff0c;我们将深入探讨供应链大数据如何帮助企业洞察市场趋势、优化库存管理、降低…

K线实战分析系列之八:十字星——容易识别的特殊形态

K线实战分析系列之八&#xff1a;十字星——容易识别的特殊形态 一、十字启明星和十字黄昏星二、弃婴底部形态和弃婴顶部形态三、总结十字启明星和十字黄昏星形态的要点 一、十字启明星和十字黄昏星 当开盘价与收盘价极为接近的时候&#xff0c;当期的K线就呈现为一根十字线&am…

PyPDF2:项目实战源码分享(PDF裁剪)

目录&#x1f4d1; 1. 背景&#x1f4d1;2. 源码模块解析&#x1f4d1;2.1 读取PDF页数2.2 获取指定页的宽高尺寸2.3 裁剪单页PDF2.4 批量裁剪PDF 总结&#x1f4d1; 1. 背景&#x1f4d1; 接PyPDF2模块推荐博文中提到的实际需求&#xff08;将银行网站下载来的多页且单页多张…

设计模式--单例模式--懒汉饿汉

单例模式 单例模式(Singleton)&#xff0c;保证一个类仅有一个实例&#xff0c;并提供一个访问它的全局访问点。 单例模式 通常我们可以让一个全局变量使得一个对象被访问&#xff0c;但它不能防止你实例化多个对象。一个最好的办法就是&#xff0c;让类自身负责保存它的唯一实…

[嵌入式系统-34]:RT-Thread -19- 新手指南:RT-Thread标准版系统架构

目录 一、RT-Thread 简介 二、RT-Thread 概述 三、许可协议 四、RT-Thread 的架构 4.1 内核层&#xff1a; 4.2 组件与服务层&#xff1a; 4.3 RT-Thread 软件包&#xff1a; 一、RT-Thread 简介 作为一名 RTOS 的初学者&#xff0c;也许你对 RT-Thread 还比较陌生。然…

vue-element-admin如何绕开系统的请求的路由,使用静态路由

我开发时候遇到一个这样的问题&#xff0c;服务端权限管理部分还没搞好&#xff0c;所以但是需要将所有菜单列出来 我做了以下的修改 首先是建一个文件存放后期需要动态生成的路由 引入到常量路由中 另外在permissions当中

FPGA之16:1复选器

每个slice 都有一个F8MUX。F8MUX原语&#xff1a; MUXF8 MUXF8_inst&#xff08; .0&#xff08;0&#xff09;&#xff0c;Il Output of MUX to general routing .I0&#xff08;10&#xff09;&#xff0c;//Input&#xff08;tie to MUXF7L/LO out&#xff09; .I1&#xf…

This dependency was not found解决方法

问题如上(前端代码)&#xff0c;我是引用js文件出的问题&#xff0c;无法找到api/userManage模块。 解决&#xff1a;没感觉哪有问题&#xff0c;把后面加了个/&#xff0c;就解决了&#xff0c;代表src目录&#xff0c;应该是目录和目录之间应该有/作为分割&#xff1a;

vue3+js 实现记住密码功能

常见的几种实现方式 1 基于spring security 的remember me 功能 ​​​​​​​ localStorage 除非主动清除localStorage 里的信息 &#xff0c;不然永远存在&#xff0c;关闭浏览器之后下次启动仍然存在 存放数据大小一般为5M 不与服务器进行交互通信 cookies 可以…

C++ //练习 8.4 编写函数,以读模式打开一个文件,将其内容读入到一个string的vector中,将每一行作为一个独立的元素存于vector中。

C Primer&#xff08;第5版&#xff09; 练习 8.4 练习 8.4 编写函数&#xff0c;以读模式打开一个文件&#xff0c;将其内容读入到一个string的vector中&#xff0c;将每一行作为一个独立的元素存于vector中。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09…

【Java程序设计】【C00314】基于Springboot的足球俱乐部管理系统(有论文)

基于Springboot的足球俱乐部管理系统&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 项目简介 这是一个基于Springboot的足球俱乐部管理系统的设计与实现&#xff0c;本系统有管理员以及用户二种角色权限 管理员权限操作的功能包括管理公告&#x…