【C++进阶(五)】STL大法--list模拟实现以及list和vector的对比

这篇具有很好参考价值的文章主要介绍了【C++进阶(五)】STL大法--list模拟实现以及list和vector的对比。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

💓博主CSDN主页:杭电码农-NEO💓

⏩专栏分类:C++从入门到精通⏪

🚚代码仓库:NEO的学习日记🚚

🌹关注我🫵带你学习C++
  🔝🔝


【C++进阶(五)】STL大法--list模拟实现以及list和vector的对比,C++从入门到精通,c++,list,java


1. 前言

本篇文章立足于上一篇文章:
list深度剖析(上)
请先阅读完上一篇文章后再阅读这篇文章!

本章重点:

本章着重讲解list的模拟实现
list模拟实现的重难点是迭代器的实现
和const迭代器的实现
最后总结list和vector的区间对比

注:我将在文章末尾分享list模式实现全部代码


2. list类的大致框架与结构

由于list类是一个带头双向循环链表
所以在写list类之前,应该先定义节点类
在真正的list类中会复用此类:

template<class T>
class list_node
{
public:
	T _data;
	list_node<T>* _next;
	list_node<T>* _prev;
	list_node(const T& x = T())
		:_data(x)
		, _next(nullptr)
		, _prev(nullptr)
	{}
};

这个类和用C语言实现链表的定义很像
节点中存储一个T模板类型的值和
上一个节点的地址和下一个节点的地址

在List类中,由于链表都是些链接关系
所以List类中的成员变量只需要定义一个
那就是头节点head,知道head的链接关系
就知道list类对象中存放的内容!

List类基本框架以及无参构造:

template<class T>
class List
{
	typedef list_node<T> node;
public:
	List()
	{
		_head = new node;
		_head->_next=_head;
		_head->_prev=_head;
	}
private:
	node* _head;
}

给头节点head开辟一份空间后
头节点的指向最开始都是指向自身:

【C++进阶(五)】STL大法--list模拟实现以及list和vector的对比,C++从入门到精通,c++,list,java


3. List类的构造,析构,拷贝构造

无参构造函数以及写过了,我们模仿
STL库中的带参构造写一个用迭代器
区间初始化的构造函数:

void emptyinit()//创建并初始化哨兵位的头节点,方便给构造和拷贝构造复用
	{
		_head = new node;
		_head->_next = _head;
		_head->_prev = _head;
	}
template<class InputTterator>//有参的构造
List(InputTterator first, InputTterator last)
{
	emptyinit();
	while (first != last)
	{
		push_back(*first);
		first++;
	}
}

注:push_back先用着,后面会实现

在实现析构函数前,我们可以先实现clear函数
因为析构函数实际上可以复用clear函数:

void clear()//清空数据,除了头节点
{
	iterator it = begin();
	while (it != end())
	{
		it = erase(it);//erase会返回下一个节点的迭代器
	}
}

~List()//_head也要被删除掉
{
	clear();
	delete _head;
	_head = nullptr;
}

注:迭代器相关的函数先用着,后面会实现

拷贝构造函数的实现:(用lt1拷贝lt2)

//lt2(lt1)
List(const List<T>& lt)//完成深拷贝
{
	emptyinit();
	List<T> tmp(lt.begin(), lt.end());//用lt1的迭代器区间去构造一下tmp
	::swap(_head, tmp._head);
}

此拷贝构造函数精妙的地方有两个:

  1. 它先定义一个临时变量tmp来接受
    lt1的所有值,然后再将临时变量tmp
    的head和lt2的head交换,这样一来
    lt2中的链接关系就和lt1相同了
    并且tmp变量是构造函数初始化的
    它是深拷贝,所以lt2对于lt1也是深拷贝
  1. tmp是临时变量,除了作用域会销毁
    也就是除了此拷贝构造函数后会销毁
    销毁时会调用析构函数,然而lt2的head
    以及和tmp的head交换了,所以tmp销毁
    时实际上是在帮原先的lt2销毁内存!

4. list的迭代器的实现

由于list的迭代器底层不是简单的指针
所以我们不能像实现vector一样直接在
list类定义迭代器和使用迭代器相关函数

要重新写一个迭代器类来实现功能:

template<class T>
struct __list_iterator
{
	typedef list_node<T> node;
	typedef __list_iterator iterator;
	node* _node;
}

这样重新写一个类的作用是将迭代器的
++,- -等操作在此类中实现,因为list中的
++实际上是node=node->next,然而list
中的==符号判断的是node1->data和
node2->data相不相同,如果迭代器和
list写在一个类中,实现此等操作会很麻烦


4.1 list迭代器的若干函数解析

1. 构造函数:

__list_iterator(node* nnode)
	:_node(nnode)
	{}

由于初始化列表会调用node的构造函数
所以list迭代器的构造函数可写可不写

2. 前置++/- -和后置++/- -

iterator& operator++()//前置++
{
	_node = _node->_next;
	return *this;
}
iterator& operator++(int)//后置++
{
	iterator tmp = *this;
	_node = _node->_next;
	return tmp;
}
iterator& operator--()//前置--
{
	_node = _node->_prev;
	return *this;
}
iterator& operator--(int)//后置--
{
	iterator tmp = *this;
	_node = _node->_prev;
	return tmp;
}

3. 等于和不等于函数解析:

bool operator!=(const iterator& it)const
{
	return _node != it._node;
}
bool operator==(const iterator& it)const
{
	return _node == it._node;
}

4.2 list迭代器的解引用和箭头操作

迭代器的使用就像指针一样,所以
解引用后应该直接得到节点的数据!

由于结构体变量除了可以用解引用
以外还能使用->,所以需要写两个函数:

//可读可写
T& operator*()
{
	return _node->_data;
}
//可读可写
T* operator->()
{
	return &(operator*());
}

解引用操作的应该没有问题
重点难点在这个箭头->函数
可以将这个函数一步一步拆分:

return &(_node->_data);

相当于返回了一个结构体指针

然而我们想要的并不是一个结构体指针
而是确切的值,蛋这样写编译器并不会报错

这是因为编译器为了代码的可读性
帮我们省略了一个箭头->
所以只需要一个箭头->就能访问数据!

注:省略的箭头可以将返回的结构体指针解引用
然而此结构体指针解引用后其实就是data


4.3 迭代器类映射到list类

当我们实现完迭代器类后
就可以在list类中复用迭代器类了:

template<class T>
class List
{
	typedef list_node<T> node;
	typedef __list_iterator<T> iterator;
private:
	node* _head;

有了迭代器类的加持后,list类中的
其他函数如begin和end就是歪瓜裂枣了

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

注:其他关于迭代器的函数会在最后放出

5. const迭代器实现深度剖析

既然list中的迭代器和vector中的不同
那么const迭代器的实现当然也不同

首先我们要明白一点,list的普通迭代器
和const迭代器的区别到底是什么?

const对象的剖析:

const迭代器是为const对象准备的
然而const对象的特征就是不能修改
那么什么操作会让对象的值变化呢?
答案很明显是解引用操作和箭头->
begin和end函数返回后也有可能被修改
注:返回值是T&和T*的函数都要注意

解决方案剖析:

大家可能第一时间想到再重新写一个
const迭代器的类,里面的实现和普通
迭代器都一样,唯一不同的是解引用函数
和箭头->函数的返回值是const对象

【C++进阶(五)】STL大法--list模拟实现以及list和vector的对比,C++从入门到精通,c++,list,java


5.1 const迭代器实现详解

首先,不用再重新写一个类来实现
接下来的方案不要问为什么
不要问怎么想到的,是天才想到的:

在普通迭代器类中使用三个模板参数

为什么要这样做?

通过观察可以发现,只有两个函数不同
并且这两个函数的类型只有T&和T*
那么就专门为这两个类型设置两个模板
只有在编写这两个函数时用到这两个模板
其余函数还是正常的使用T来当类型

话不多说,上代码

template<class T, class Ref, class Ptr>
struct __list_iterator//迭代器不需要析函数
{
	typedef list_node<T> node;
	typedef __list_iterator iterator;
	node* _node;
}

模板中的Ref代表的是引用&
模板中的Ptr代表的是指针
*

现在重新来实现一下这两个函数:

//按模板参数来
Ref operator*()
{
	return _node->_data;
}
Ptr operator->()
{
	return &(operator*());
}

现在你脑子肯定是一篇空白
但是精髓的地方马上就来了,请耐住性子


5.2 const迭代器和list类的复用

当list类复用了此迭代器类后:

template<class T>
class List
{
	typedef list_node<T> node;
	typedef __list_iterator<T, T&, T*> iterator;
	typedef __list_iterator<T, const T&, const T*> const_iterator;
private:
	node* _head;
}

这样写出来后,普通迭代器和const迭代器
就被完美的区别开了,当传入const对象时
系统会把模板参数实例化为const T&
当传入的是普通对象时,系统会把模板参数
实例化为普通的T,T&和T*

begin和end函数的复写:

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

5.3 const迭代器使用实例

现在,我们通过一个实例来感受这一过程:

void print_list(const list<int>& lt)
{
	auto it = lt.begin();
	while (it != lt.end())
	{
		//*it = 10;
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

此时的lt变量是const变量,实例化类时
会实例化为<T,const T&,const T*>
然后回退到迭代器类时,迭代器类的
模板参数Ref就对应:const T&
模板参数Ptr就对应:const T*

此时解引用函数的返回值就是const T&
如果你写:*it=10;那么就会报错!


6. list和vector的对比

详情请看下表:

目录 vector list
底层结构 顺序表,空间连续 带头双向循环链表
随机访问 支持随机访问,访问效率为O(1) 不支持随机访问,访问效率为O(N)
插入和删除 任一位置插入删除效率低,还需扩容 任一位置插入效率高
空间利用率 空间,缓存利用率高 不连续空间,容易造成内存碎片
迭代器 原生态的指针 对原生指针的封装
迭代器失效 插入和删除都会导致 只有删除会导致
使用场景 高效存储,支持随机访问 有大量插入和删除操作,不关心随机访问

注:这个表格不能死记硬背,要理解记忆


7. 总结以及代码分享

总的来说,list的底层实现较于vector
来说要复杂一点,这其中的底层原因
就是list的迭代器还需要一层封装,而
vector的迭代器不需要额外封装

C++的强大就在于把复杂的底层
全部封装起来了,而表面的使用上
list和vector并无太大区别,这就是
C++封装的魅力!

list模拟实现全部代码分享:

List模拟实现全部代码

注:全部代码中包含反向迭代器
目前阶段可以跳过不管文章来源地址https://www.toymoban.com/news/detail-701759.html


🔎 下期预告:栈和队列 🔍

到了这里,关于【C++进阶(五)】STL大法--list模拟实现以及list和vector的对比的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 【C++进阶(一)】STL大法以及string的使用

    【C++进阶(一)】STL大法以及string的使用

    💓博主CSDN主页:杭电码农-NEO💓   ⏩专栏分类:C++从入门到精通⏪   🚚代码仓库:NEO的学习日记🚚   🌹关注我🫵带你学习C++   🔝🔝 由于C语言的 标准库不够强大 没有数据结构和一些基本算法 什么都需要程序员自己实现 所以C语言在某种意义上并不实用 本章重点: 本章会

    2024年02月11日
    浏览(10)
  • C++ STL vector 模拟实现

    C++ STL vector 模拟实现

    ✅1主页:我的代码爱吃辣 📃2知识讲解:C++之STL 🔥3创作者:我的代码爱吃辣 ☂️4开发环境:Visual Studio 2022 💬5前言:上次我们已经数字会用了vector,这次我们对其底层更深一步挖掘,其中重点是,Vector中一些深浅拷贝问题。 目录 一.Vector模拟实现的整体框架 二. Vector的构

    2024年02月13日
    浏览(14)
  • 【C++ STL】vector模拟实现

    【C++ STL】vector模拟实现

    2023年05月17日
    浏览(27)
  • C++ [STL之vector模拟实现]

    C++ [STL之vector模拟实现]

    本文已收录至《C++语言》专栏! 作者:ARMCSKGT vector是STL容器容器之一,其底层实现类似于数据结构顺序表,相当于string来说得益于泛型模板的加持使得vector可以变为任何类型,且是可以动态扩容,堪称大号数组!在vector的实现中,有许多值得我们学习的细节,接下来将为大家

    2024年02月11日
    浏览(6)
  • C++ —— STL容器【vector】模拟实现

    C++ —— STL容器【vector】模拟实现

    本章代码gitee仓库:vector模拟实现、vector源码 看源码发现 vector 是类模板定义的,成员是采用迭代器进行管理 当涉及到容器类时,通常有一些关键函数,如构造函数、析构函数和拷贝构造函数,它们负责初始化容器对象、销毁对象和进行对象的拷贝等 这里注意拷贝构造要实现

    2024年02月16日
    浏览(8)
  • 【C++】STL 模拟实现之 vector

    【C++】STL 模拟实现之 vector

    vector 是我们学习的第一个真正的 STL 容器,它接口的使用方式和 string 有一点点的不同,但大部分都是一样的,所以这里我们就只演示其中一些接口的使用,大家如果有疑惑的地方直接在 cplusplus 是上面查看对应的文档即可。 vector 提供了四种构造方式 – 无参构造、n 个 val 构

    2023年04月27日
    浏览(10)
  • STL 关于vector的细节,vector模拟实现【C++】

    STL 关于vector的细节,vector模拟实现【C++】

    _start指向容器的头,_finish指向容器当中 有效数据 的下一个位置,_endofstorage指向整个容器的尾 先开辟一块与该容器大小相同的空间,然后将该容器当中的数据一个个拷贝过来即可,最后更新_finish和_endofstorage的值即可。 深拷贝版本一: 注意: 不能使用memcpy函数 , 如果vec

    2024年02月15日
    浏览(9)
  • 【c++】:STL中vector的模拟使用及模拟实现

    【c++】:STL中vector的模拟使用及模拟实现

        文章目录 前言 一.使用库中vector常用接口 二.vector的模拟实现 总结   上一篇我们讲解了STL中的string的使用和模拟实现,这次我们就来讲解STL中的vector,vector相对于string来说模拟实现会难一些,难点在于迭代器失效问题和深浅拷贝问题。 首先介绍一下vector: 1. vector是表示

    2024年01月21日
    浏览(10)
  • C++ stl容器vector的底层模拟实现

    C++ stl容器vector的底层模拟实现

    目录 前言:   1.成员变量,容量与大小 2.构造函数 无参构造: 带参的使用值进行构造:  使用迭代器区间进行构造: 3.交换 4.拷贝构造 5.赋值重载 6.迭代器 7.扩容 reserve: resize: 8.插入与删除 insert: erase: insert迭代器失效问题: erase迭代器失效问题: 9.头插头删 10.[]重载

    2024年04月15日
    浏览(9)
  • 【C++】STL之vector功能及模拟实现

    【C++】STL之vector功能及模拟实现

    目录 前沿 一、vector的使用  1、vector 构造函数的声明  2、vector 迭代器的使用  3、vector 空间增长问题  4、vector 的增删查改 二、vector的模拟实现  1、vector 的成员变量  2、迭代器  3、容量相关(resize, reserve)  4、数据访问相关  5、插入删除   5.1 任意位置插入   5.2 任意位置

    2024年02月16日
    浏览(5)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包