STL容器之list

​ 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;

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

[嵌入式系统-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 还比较陌生。然…

*MYSQL--索引--内部原理

MYSQL的索引根据功能,主要有三大类型: 1.HASH索引 2.二叉树 3.BTREE索引 一:HASH索引 1.内部原理: 在设置了某列为索引列之后,并且开始或者将要在相应索引列创建数据的时候,系统通过某种算法 F(X) 自动计算出来一个十六进制的哈希值,这个哈希值能够对应相应的字段值 所以…

2.openEuler概述及安装指南(二)

openEuler OECA认证辅导,标红的文字为学习重点和考点。 如果需要做实验,建议下载麒麟信安、银河麒麟、统信等具有图形化的操作系统,其安装与openeuler基本一致。 1.安装过程及配置 使用光盘引导安装: 此处以光盘安装为例介绍安装openEuler,其他安装方式除在启动安装时的…

我的NPI项目之设备系统启动(八) -- Android14的GKI2.0开发步骤和注意事项

GKI是什么&#xff1f; Google为什么要推行GKI&#xff1f; GKI全称General Kernel Image。GKI在framework和kernel之间提供了标准接口&#xff0c;使得android OS能够轻松适配/维护/兼容不同的设备和linux kernel。 Google引入GKI的目的是将Framework和Kernel进一步的解耦。因…

virtualenv env_name 使用 virtualenv 创建 python 虚拟环境

为什么要用这个 win7 32 环境下 pycharm 只能用低版本的&#xff0c;比如 2016,2018 此时pycharm 图形界面创建的 虚拟环境版本很低&#xff0c;有些包不兼容&#xff0c;因此用 virtualenv 模块&#xff0c;可以创建 20 版本以上的虚拟环境 virtualenv env_name官方文档 http…

java_URL中的URL编码转换成中文

问题描述 上传文件后&#xff0c;获得的URL中包含了URL编码&#xff0c;导致在前端展示文件名时出现乱码 最终效果 解决思路&#xff1a; 1、先按照英文逗号切割URL 2、截取字符串中URL编码部分(含后缀名) 3、使用正则匹配截取到的字符串中的URL编码 4、转换URL编码为中文&a…

React18源码: Fiber树中的全局状态与双缓冲

Fiber树构造 在React运行时中&#xff0c;fiber树构造位于 react-reconciler 包在正式解读 fiber 树构造之前&#xff0c;再次回顾一下renconciler的4个阶段 1.输入阶段&#xff1a;衔接react-dom包&#xff0c;承接fiber更新请求2.注册调度任务&#xff1a;与调度中心(schedu…

HarmonyOS创建一个ArkTS卡片

创建一个ArkTS卡片 在已有的应用工程中&#xff0c;创建ArkTS卡片&#xff0c;具体操作方式如下。 创建卡片。 根据实际业务场景&#xff0c;选择一个卡片模板。 在选择卡片的开发语言类型&#xff08;Language&#xff09;时&#xff0c;选择ArkTS选项&#xff0c;然后单…

QT Widget自定义菜单

此文以设置QListWidget的自定义菜单为例&#xff0c;其他继承于QWidget的类也都可以按类似的方法去实现。 1、ui文件设置contextMenuPolicy属性为CustomContextMenu 2、添加槽函数 /*** brief onCustomContextMenuRequested 右键弹出菜单* param pos 右键的坐标*/void onCusto…

2024如何恢复旧版的Chrome的主题样式

起因 chrome 更新版本之后的主题样式变成了浅紫色的页签卡样式&#xff0c;感觉很不习惯&#xff0c;也很不喜欢 如何换回旧版主题 通过主题商店&#xff0c;安装旧版本的主题 主题商店搜索下面&#xff0c;或着直接访问下面的地址 Chrome Original White Theme https://…

基于SpringBoot的家教管理系统

基于SpringBootVue的家教管理系统的设计与实现~ 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringBootMyBatis工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 前台主页 家教 个人中心 管理员界面 摘要 本文介绍了基于SpringBoot框架开发的家…

神经网络系列---分类度量

文章目录 分类度量混淆矩阵&#xff08;Confusion Matrix&#xff09;&#xff1a;二分类问题二分类代码多分类问题多分类宏平均法:多分类代码多分类微平均法&#xff1a; 准确率&#xff08;Accuracy&#xff09;&#xff1a;精确率&#xff08;Precision&#xff09;&#xf…

模型评估方式

文章目录 一、有监督-分类模型1、混淆矩阵2、分类模型的精度和召回率3、ROC曲线与AUC 二、有监督-回归模型1、均方误差MSE2、 R 2 R^2 R2决定系数3、回归模型代码示例 三、无监督模型1、kmeans求解最优k值的方法&#xff1a;轮廓系数、肘部法2、GMM的最优组件个数&#xff1a;A…

【Vue3】‘vite‘ 不是内部或外部命令,也不是可运行的程序或批处理文件。

问题 今天拿到别人项目的时候&#xff0c;我平时比较习惯用pnpm&#xff0c;我就使用pnpm i先下载依赖包&#xff0c;下载完成后&#xff0c;启动项目&#xff0c;就开始报以下错误&#xff01; 但是当我执行pnpm i的时候&#xff0c;vite不应该就已经被我下载下来了吗 研究了…

线程共享和非共享的资源及线程优缺点

注意&#xff1a;共享的内存地址空间中不包括栈&#xff1b;共享文件描述符表&#xff0c;表示&#xff0c;同一进程中线程可以操作同一文件。

【机器学习】特征工程之特征选择

&#x1f388;个人主页&#xff1a;豌豆射手^ &#x1f389;欢迎 &#x1f44d;点赞✍评论⭐收藏 &#x1f917;收录专栏&#xff1a;机器学习 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共同学习、交流进…

【ubuntu】永久修改主机名

文章目录 1. 问题描述2. 解决方案 1. 问题描述 主机名过长&#xff08;后面的部分&#xff09; 2. 解决方案 查看主机名详情 hostnamectl修改指定主机名 hostnamectl set-hostname ubuntu2204 --static登出重进即可

基于java+springboot+vue实现的美食信息推荐系统(文末源码+Lw)23-170

1 摘 要 使用旧方法对美食信息推荐系统的信息进行系统化管理已经不再让人们信赖了&#xff0c;把现在的网络信息技术运用在美食信息推荐系统的管理上面可以解决许多信息管理上面的难题&#xff0c;比如处理数据时间很长&#xff0c;数据存在错误不能及时纠正等问题。这次开发…

虚拟机安装Docker装载Mysql

目录 1.安装docker 2. docker中安装mysql 1.选择mysql镜像 2.查看镜像 3.启动mysql 4.修改配置 5.进入容器查看配置&#xff1a; 6.设置启动docker时&#xff0c;即运行mysql 1.安装docker SSH 登录到虚拟机: 使用MobaXterm或其他SSH客户端连接到虚拟机&#xff1a; ss…

前后端延迟怎么解决

当今互联网应用的发展越来越迅猛&#xff0c;用户对于网站或应用的性能要求也越来越高。其中一个重要方面就是前后端延迟的解决&#xff0c;也就是减少前端与后端之间的通信时间延迟&#xff0c;提高用户体验。本文将详细介绍如何解决前后端延迟的问题。 网络延迟 数据在网络…