【C++/STL】:list容器的深度剖析及模拟实现

目录

  • 🚀前言
  • 🚀一,节点类
  • 🚀二,迭代器类
    • 1,普通迭代器类的实现
    • 2,->运算符的使用场景
    • 3,const迭代器类的实现
    • 4,通过模板参数,把两个类型的迭代器类结合
    • 5,迭代器类的一些问题的思考
  • 🚀三,list 类
    • 1,list类的结构
    • 2,迭代器的实现
    • 3,插入数据insert
    • 4,删除数据erase
    • 5,头插,头删,尾插,尾删
    • 6,常见构造函数的实现
    • 7,析构函数

🚀前言

点击跳转到文章:【list的基本使用】
要模拟实现list,必须要熟悉list的底层结构以及其接口的含义,list的底层是带头双向循环链表,通过上一篇文章的学习,这些内容已基本掌握,现在我们来模拟实现list容器的主要接口。

与前面的vector类似,由于使用了模板,也只分成.cpp和.h两个文件。

.cpp文件里放节点类,迭代器类,list类及其成员函数,测试函数的实现,在.h文件里进行测试

本文的重点是:对三个类的区分与理解,迭代器类的实现

🚀一,节点类

1.为什么定义节点结构体时使用struct而不是class?

答:(1)其实用class也可以,但是class与struct默认的访问限定不同,当没有声明公有,私有时,struct内容默认是公有,class内容默认的私有,所以用class要加上public

(2)当我们用class没有加上public,也没有实例化对象时,编译不会报错(报私有成员的错误),因为模版是不会被细节编译的。只有当我们实例化出对象,模版才会被编译,并且类的实例化并不是对所有成员函数都实例化,而是调用哪个成员函数就实例化哪个。这叫做按需实例化

2.可用匿名对象初始化。如果T是自定义类型,则调用其默认构造,并且T是内置类型也升级成了有默认构造的概念了。

template <class T>
struct ListNode
{ListNode<T>* _next;ListNode<T>* _prev;T _data;ListNode(const T& data = T()):_next(nullptr),_prev(nullptr),_data(data){}
};

🚀二,迭代器类

前面学习的string类和vector的迭代器用的是原生指针类型,即T*。但是在list容器中是不能这样的,因为前面两者的底层物理空间是连续的,符合迭代器++与- -的行为。但是list是由一个一个节点构成的,物理空间不连续,Node*的++和- -不符合迭代器的行为,无法变遍历

所以用一个类把Node* 封装,就可以重载运算符,使得用起来像内置类型,但会转换成函数调用,继而控制Node*的行为

1,普通迭代器类的实现

遍历需要的核心运算符重载是 *,!=,++ 和 ->。所以只需要利用带头双向循环链表的特性,对Node * 进行封装,从而控制Node * 的行为。

class ListIterator
{typedef ListNode<T> Node;typedef ListIterator<T> Self;//名字变得简短public:Node* _node;//定义一个节点指针ListIterator(Node* node):_node(node){}//前置:返回之后的值//++it;//返回与自己一样的类型Self& operator++(){_node = _node->_next;return *this;}Self& operator--(){_node = _node->_prev;return *this;}//后置:返回之前的值Self operator++(int){Self tmp(*this);_node = _node->_next;return tmp;}Self operator--(int){Self tmp(*this);_node = _node->_prev;return tmp;}T& operator*(){return _node->_data;}//返回的是数据的地址T* operator->(){return &_node->_data;}bool operator!=(const Self& it){return _node != it._node;}bool operator==(const Self& it){return _node == it._node;}
};

2,->运算符的使用场景

假设某个场景下存在一个坐标类:

struct Pos
{int _row;int _col;Pos(int row = 0,int col = 0):_row(row),_col(col){}
};

如果我们插入坐标,并且想要打印出坐标,该如何遍历?

错误示范

void test_list2()
{list<Pos> lt1;lt1.push_back(Pos(100, 100));//使用匿名对象lt1.push_back(Pos(200, 200));lt1.push_back(Pos(300, 300));//这里的it是Pos*,是结构体指针list<Pos>::iterator it = lt1.begin();while (it != lt1.end()){cout << *it << " ";//err++it;}cout << endl;
}

原因:因为这里的*it返回的是Pos自定义类型,而访问自定义类型需要需要在类中自己重载流插入(<<),这里并没有重载,所以报错

正确操遍历的两种方式

方式1:通过.操作符直接访问结构体的成员变量(一般不这样访问数据)。

cout << (*it)._row << ":" << (*it)._col << endl;//ok

方式2:通过重载->运算符,对结构体指针进行解引用。

cout << it.operator->()->_row << ":" << it.operator->()->_col << endl;//ok

注意:其实这里严格来说是有两个箭头,第一个运算符重载的调用 it.operator->() 返回的是 Pos*,第二个箭头才是原生指针,Pos*再用箭头访问。为了可读性,省略了一个->

void test_list2()
{list<Pos> lt1;lt1.push_back(Pos(100, 100));//使用匿名对象lt1.push_back(Pos(200, 200));lt1.push_back(Pos(300, 300));//这里的it是Pos*,是结构体指针list<Pos>::iterator it = lt1.begin();while (it != lt1.end()){	//方式1://cout << (*it)._row << ":" << (*it)._col << endl;//ok//*it就是Pos结构体,再用.操作符访问成员//方式2:cout << it->_row << ":" << it->_col << endl;//ok//cout << it.operator->()->_row << ":" << it.operator->()->_col << endl;//ok++it;}cout << endl;
}

3,const迭代器类的实现

在我们遍历数据时,有时会写一个打印函数,引用传参,一般建议加const,这就出现了一个const链表

void Func(const list<int>& lt1)
{list<int>::const_iterator it = lt1.begin();while (it != lt1.end()){cout << *it << " ";++it;}cout << endl;
}

const迭代器不是在普通迭代器前面加const,即不是const iterator

//err 这样使it本身也不能++了
const list< int >::iterator it = it.begin();

const 迭代器目的:本身可以修改,指向的内容不能修改,类似const T* p。

所以我们要再定义一个类,控制*和->的返回值就可以了。

template <class T>
class ListConstIterator
{typedef ListNode<T> Node;typedef ListConstIterator<T> Self;//名字变得简短public:Node* _node;//定义一个节点指针ListConstIterator(Node* node):_node(node){}//前置:返回之后的值//++it;//返回与自己一样的类型Self& operator++(){_node = _node->_next;return *this;}Self& operator--(){_node = _node->_prev;return *this;}//后置:返回之前的值Self operator++(int){Self tmp(*this);_node = _node->_next;return tmp;}Self operator--(int){Self tmp(*this);_node = _node->_prev;return tmp;}// 所以我们要再定义一个类,使用const控制*和->的返回值就可以const T& operator*(){return _node->_data;}const T* operator->(){return &_node->_data;}bool operator!=(const Self& it){return _node != it._node;}bool operator==(const Self& it){return _node == it._node;}
};

4,通过模板参数,把两个类型的迭代器类结合

可以发现,其实普通迭代器和const迭代器的本质区别是 * 和 ->,这两个运算符的返回类型的变化。两个类冗余,所以可以通过模板,给不同的模板参数,让编译器自己实例化两个类

template <class T,class Ref,class Ptr>
struct ListIterator
{typedef ListNode<T> Node;typedef ListIterator<T, Ref, Ptr> Self;//名字变得简短Node* _node;//定义一个节点指针ListIterator(Node* node):_node(node){}//前置:返回之后的值//++it;//返回与自己一样的类型Self& operator++(){_node = _node->_next;return *this;}Self& operator--(){_node = _node->_prev;return *this;}//后置:返回之前的值Self operator++(int){Self tmp(*this);_node = _node->_next;return tmp;}Self operator--(int){Self tmp(*this);_node = _node->_prev;return tmp;}Ref operator*(){return _node->_data;}Ptr operator->(){return &_node->_data;}bool operator!=(const Self& it){return _node != it._node;}bool operator==(const Self& it){return _node == it._node;}
};

5,迭代器类的一些问题的思考

(1) 类中是否需要写析构函数

这个迭代器类不要写析构函数,因为这里的节点不是迭代器的,是链表的,不用把它释放。我们使用begin,end返回节点给迭代器,是借助迭代器修改,访问数据,所以我们不需要释放

(2) 类中是否需要写拷贝构造进行深拷贝和写赋值拷贝

这里也不需要写拷贝构造进行深拷贝,因为这里要的就是浅拷贝。begin返回了第一个节点的迭代器给it,这里就是用默认生成的拷贝构造,浅拷贝给it,那这两个迭代器就指向同一个节点,所以这里用默认的拷贝构造和赋值拷贝就可以了

🚀三,list 类

1,list类的结构

template <class T>
class list
{typedef ListNode<T> Node;
public://物理空间不是连续的,不符合迭代器的行为,无法遍历//typedef Node* iterator;//规范命名//typedef ListIterator<T> iterator;//typedef ListConstIterator<T> const_iterator;typedef ListIterator<T, T&, T*> iterator;typedef ListIterator<T, const T&, const T*> const_iterator;//………………private:Node* _head;
};

2,迭代器的实现

包含普通迭代器和const迭代器

iterator begin()
{//iterator it(_head->_next);//return it;return iterator(_head->_next);//使用匿名对象
}iterator end()
{return iterator(_head);
}const_iterator begin()const
{return const_iterator(_head->_next);
}const_iterator end()const
{return const_iterator(_head);
}

3,插入数据insert

iterator insert(iterator pos, const T& x)
{Node* cur = pos._node;//找到当前节点Node* newnode = new Node(x);//申请节点Node* prev = cur->_prev;//找到前一个节点//prev newnode cur 进行链接newnode->_next = cur;cur->_prev = newnode;prev->_next = newnode;newnode->_prev = prev;return iterator(newnode);
}

注意:链表的insert没有迭代器失效问题,因为没有扩容的概念,pos位置的节点不会改变。但是STL库里insert也给了返回值,返回的是新插入位置的迭代器

4,删除数据erase

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;return iterator(next);
}

注意:链表的erase后有迭代器失效问题,pos失效了,因为pos指向的节点被释放了。所以也要返回值,返回的是删除节点的下一个节点的迭代器

5,头插,头删,尾插,尾删

可以复用前面的 insert和 erase

//尾插:end()的下一个位置
void push_back(const T& x)
{//Node* newnode = new Node(x);//申请节点并且初始化//Node* tail = _head->_prev;链接节点//tail->_next = newnode;//newnode->_prev = tail;//_head->_prev = newnode;//newnode->_next = _head;insert(end(), x);
}//尾删
void pop_back()
{erase(--end());//注意:前置--
}//头插:在begin前面插入
void push_front(const T& x)
{insert(begin(), x);
}//头删
void pop_front()
{erase(begin());
}

6,常见构造函数的实现

主要包含:构造函数,拷贝构造,initializer_list构造(列表构造)

注意:由于这些都是在有哨兵位节点的前提下实现的,所以可以把申请哨兵位头节点这一步骤提取出来。

//空初始化,申请哨兵位头节点
void empty_init()
{_head = new Node();_head->_next = _head;_head->_prev = _head;
}list()
{empty_init();
}//拷贝构造:直接复用尾插,前提要有哨兵位头节点
//lt2(lt1)
list(const list<T>& lt)
{empty_init();//注意:使用范围for时加上const和&for (const auto& e : lt){push_back(e);}
}//initializer_list构造,前提要有哨兵位头节点
list(initializer_list<T> il)
{empty_init();for (const auto& e : il){push_back(e);}
}

7,析构函数

析构函数的作用是:删除整个链表结构,包括哨兵位节点

//清空当前数据 留头节点,其余节点释放
void clear()
{auto it = begin();while (it != end()){//返回删除节点的下一个节点的迭代器it = erase(it);}
}//析构:销毁整个链表
~list()
{clear();delete _head;_head = nullptr;
}

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

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

相关文章

基于springboot+vue的梦幻玩具乐园的设计与实现(在线购物平台)

需要源码和论文的小伙伴可以私信博主&#xff08;有偿&#xff09; ​​​​​课题目的与意义 随着互联网的不断普及与在线销售平台的迅猛发展&#xff0c;在线购物日益受到广大消费者的青睐与追捧。通过构建基于Spring BootVue的在线玩具商城&#xff0c;可以为玩具制造商、…

如何快速交付网络基础设施运维管理软件项目?

​ 基于nVisual网络基础设施数字孪生管理工具 开发项目需求 项目交付成本节省50%、进度提高100% ​ &#xff1e;&#xff1e;&#xff1e;nVisual主要功能&#xff1c;&#xff1c;&#xff1c; 01 场 景 ★ 支持层次化的场景结构 ★ 支持多种空间场景 ​ 02 规 划 ★ 丰…

基于Pytorch框架的深度学习ConvNext神经网络宠物猫识别分类系统源码

第一步&#xff1a;准备数据 12种宠物猫类数据&#xff1a;self.class_indict ["阿比西尼猫", "豹猫", "伯曼猫", "孟买猫", "英国短毛猫", "埃及猫", "缅因猫", "波斯猫", "布偶猫&q…

Go语言之函数和方法

个人网站&#xff1a; http://hardyfish.top/ 免费书籍分享&#xff1a; 资料链接&#xff1a;https://url81.ctfile.com/d/57345181-61545511-81795b?p3899 访问密码&#xff1a;3899 免费专栏分享&#xff1a; 资料链接&#xff1a;https://url81.ctfile.com/d/57345181-6…

学习TS看这一篇就够了!

目录 TS的优点和缺点基础类型数字类型布尔类型字符串类型void 类型null 类型和 undefined 类型bigint类型Symbol类型 其他类型数组元组枚举Enum对象和函数any void never unknown 的区别是什么泛型 Generic交叉类型联合类型 特殊符号 ? ?. ?? ! _修饰符 TS的优点和缺点 优…

GPT对话代码库——STM32G431微秒(us)级delay函数

目录 1&#xff0c;问&#xff1a; 1&#xff0c;答&#xff1a; 方法一&#xff1a;使用定时器&#xff08;Timer&#xff09; 方法二&#xff1a;使用SysTick定时器 方法三&#xff1a;使用内联汇编 选择合适的方法 2&#xff0c;问&#xff1a; 2&#xff0c;答&…

如何集成CppCheck到visual studio中

1.CPPCheck安装 在Cppcheck官方网站下载最新版本1.70&#xff0c;官网链接&#xff1a;http://cppcheck.sourceforge.net/ 安装Cppcheck 2.集成步骤 打开VS&#xff0c;菜单栏工具->外部工具->添加&#xff0c;按照下图设置&#xff0c;记得勾选“使用输出窗口” 2.…

深入解析 IPython 命名空间与作用域机制

IPython 是一个强大的交互式 Python 解释器&#xff0c;它提供了许多增强的功能来改善用户的编程体验。在 IPython 中&#xff0c;命名空间&#xff08;namespace&#xff09;和作用域&#xff08;scope&#xff09;的概念对于理解变量的生命周期和访问方式至关重要。本文将详细…

word2016中新建页面显示出来的页面没有页眉页脚,只显示正文部分。解决办法

问题描述&#xff1a;word2016中新建页面显示出来的页面没有页眉页脚&#xff0c;只显示正文部分。设置了页边距也不管用。 如图1 图1 解决&#xff1a; 点击“视图”——“多页”——“单页”&#xff0c;即可。如图2操作 图2 结果展示&#xff1a;如图3 图3

【Unity】数据持久化--JSON

1、JSON基础语法 1.1 注释内容 单行注释 // 多行注释 /* 内容 */ //注释内容 /* 多行注释 123 e1 ds */ /* 1.2 符号含义 大括号 {} 对象 中括号 [] 数组 冒号 : 键值对对应关系 逗号 , 数据分割 双引号 "" 键名/字符串 1.3 键值对表示 “ "键…

AI 大模型之美 | 更新完结

AI 大模型&#xff1a;技术的壮丽与美感 在当今的人工智能领域&#xff0c;大模型如同一座座巨大的桥梁&#xff0c;将计算机科学、语言学、认知科学等多个领域连接在一起。它们不仅仅是技术的象征&#xff0c;更是人类智慧与创新的结晶。本文将探讨AI大模型的壮丽与美感&…

基于Java的订餐小程序【附源码】

一、本选题的依据&#xff08;阐述所选课题的研究背景、研究目的和意义、分析国内外研究现状及趋势&#xff09; 研究背景&#xff1a; 随着移动互联网的普及和智能手机的发展&#xff0c;人们的生活方式正在发生深刻的变化。特别是在餐饮行业&#xff0c;传统的堂食模式已不能…

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

&#x1f4da;博客主页&#xff1a;knighthood2001 ✨公众号&#xff1a;认知up吧 &#xff08;目前正在带领大家一起提升认知&#xff0c;感兴趣可以来围观一下&#xff09; &#x1f383;知识星球&#xff1a;【认知up吧|成长|副业】介绍 ❤️如遇文章付费&#xff0c;可先看…

【深度学习】快速入门KerasNLP:微调BERT模型完成电影评论情感分类任务

简介&#xff1a;本文将介绍 KerasNLP 的安装及使用&#xff0c;以及如何使用它在情感分析任务中微调 BERT 的预训练模型。 1. KerasNLP库 KerasNLP 是一个自然语言处理库&#xff0c;兼容 TensorFlow、JAX 和 PyTorch 等多种深度学习框架。基于 Keras 3 构建&#xff0c;这些…

核密度估计kde的本质

核密度估计的本质就是插值&#xff0c;不是拟合&#xff0c;只是不要求必须过已知点。 核为box窗函数 核为高斯函数

python利用cartopy绘制带有经纬度的地图

参考&#xff1a; https://makersportal.com/blog/2020/4/24/geographic-visualizations-in-python-with-cartopy https://scitools.org.uk/cartopy/docs/latest/ https://stackoverflow.com/questions/69465435/cartopy-show-tick-marks-of-axes 具体实现方式&#xff1a; …

201.回溯算法:全排列(力扣)

class Solution { public:vector<int> res; // 用于存储当前排列组合vector<vector<int>> result; // 用于存储所有的排列组合void backtracing(vector<int>& nums, vector<bool>& used) {// 如果当前排列组合的长度等于 nums 的长度&am…

【Android】软键盘空白问题

问题描述 A界面弹出软键盘&#xff0c;跳到B界面&#xff0c;然后返回A界面时软键盘出现空白 解决方案&#xff1a; A界面的onResume方法、跳B界面方法调用前&#xff0c;加一个清除输入框焦点的方法 if (editText!null){editText.clearFocus();}清单文件里A界面添加属性&…

Mybatis 到 MyBatisPlus

Mybatis 到 MyBatisPlus Mybatis MyBatis&#xff08;官网&#xff1a;https://mybatis.org/mybatis-3/zh/index.html &#xff09;是一款优秀的 持久层 &#xff08;ORM&#xff09;框架&#xff0c;用于简化JDBC的开发。是 Apache的一个开源项目iBatis&#xff0c;2010年这…

[亲测可用] 一行代码分页---springBoot PageHelper 不生效解决方案!!

今天做mybatis查询的时候 不管是用框架查询sql还是手动写sql&#xff0c;查询分页都不生效&#xff0c;很简单&#xff0c;你配置不对&#xff0c;或者缺少了配置。我下面是直接配置 不用写代码配置。框架查询sql还是手动写sql都支持 这是我查询的sql (注意&#xff01;&#…