【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的使用

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

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

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

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

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

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

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

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

    2023年04月27日
    浏览(44)
  • C++ —— STL容器【vector】模拟实现

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

    2024年02月16日
    浏览(44)
  • STL 关于vector的细节,vector模拟实现【C++】

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

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

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

    2024年01月21日
    浏览(42)
  • C++ STL学习之【vector的模拟实现】

    ✨个人主页: 北 海 🎉所属专栏: C++修行之路 🎊每篇一句: 图片来源 The power of imagination makes us infinite. 想象力的力量使我们无限。 vector 是 STL 中的容器之一,其使用方法类似于数据结构中的 顺序表 ,得益于范型编程和 C++ 特性的加持, vector 更强大、更全能;在模拟实现

    2023年04月08日
    浏览(63)
  • C++ stl容器vector的底层模拟实现

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

    2024年04月15日
    浏览(40)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包