【C++】list原理讲解及其实现

目录
一、认识list底层结构
二、list的构造类函数
三、迭代器
四、数据的访问
五、容量相关的函数
六、关于数据的增删查改操作

前言

要模拟实现list,必须要熟悉list的底层结构以及其接口的含义,在上一篇我们仔细讲解了list的常见接口的使用及其含义,这篇我们就直接进入主题


一、list底层结构

list底层实现的是带头双向循环链表,list底层实现需要三个类,分别是链表的结点类,链表的迭代器类,链表本身

// List的节点类
template<class T>
struct ListNode
{
	ListNode<T>* _prev;
	ListNode<T>* _next;
	T _data;

	ListNode(const T& data=T())
		:_prev(nullptr)
		,_next(nullptr)
		,_data(data)
	{}
};
//List的迭代器类
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)
    {}
    Ref operator*();
    Ptr operator->();
    Self& operator++();
    Self operator++(int);
    Self& operator--();
    Self operator--(int);
    bool operator!=(const Self& l);
    bool operator==(const Self& l);
};
 //list类
template<class T>
class list
{
    typedef ListNode<T> Node;
    Node* _head;//哨兵位的头节点
    size_t _size;//链表数据个数
public:
    typedef ListIterator<T, T&, T*> iterator;
    typedef ListIterator<T, const T&, const T&> const_iterator;
}

给大家讲一下这三个类的关联,链表类是主类,链表的结点用一个类封装起来,构造起来更方便,放在主类里面不方便且冗余,由于链表的物理结构不连续,所以不能像vector和string那样单纯的用原生指针来实现,我们可以把结点类再次进行封装,封装成迭代器,让它能很好的指向链表的结点,利用它的结构优势来重载运算符遍历这个链表


二、初始化list的构造函数

1、默认构造
list();

void empty_init()
{
	_head = new Node;
	_head->_next = _head;
	_head->_prev = _head;
	_size = 0;
}

list()
{
	empty_init();
	//这里没有直接写默认构造,而是通过empty_init来实现,因为后面我们要多次用到这个函数来初始化哨兵位的头节点,写在默认构造里面就不方便其他地方的调用了
}

2、用n个val来构造链表
list(size_t n, const T& val = T());

list(size_t n, const T& val = T())
{
	empty_init();//要记得初始化哨兵位的头节点
	for (size_t i = 0; i < n; i++)
	{
		push_back(val);
	}
	_size = n;
}

3、迭代器区间构造
template <class InputIterator> list(InputIteratorfirst, InputIterator last);

list(InputIterator first, InputIterator last)
{
	empty_init();
	while (first != last)
	{
		push_back(*first);
		first++;
	}
}

特别注意:
如果你写了迭代器区间构造,就一定要重载上面的list(size_t n, const T& val = T()),如果你不重载的话,他会报一个错误 非法间接寻址,博主我深受其害🤡,再重载一个这个函数list(int n, const T& val = T());就可以 其实就是改一个类型的事

为什么要重载一下?

list<int> li(10,1);
你传的参数都是int类型,两个类型是一样的,而迭代器区间构造的类型是一样的,最符合,这就导致你没走你
想走的那个构造函数,你想走的那个函数list(size_t n, const T& val = T())在这个案例中是size_t和int类型,虽然
也可以走这个函数,但是编译器觉得迭代器区间构造更好,所以你得重载一个int,int类型的,这样编译器就会
走你重载的函数了

4、拷贝构造
(用来初始化一个正在创建的对象)
list(const list<T>& li);

list(const list<T>& li)
{
	empty_init();
	for (auto& e : li)
	{
		push_back(e);
	}
	_size = li._size;
}

5、赋值构造
(两个已经存在的对象,一个赋值给另一个)
list<T>& operator=(const list<T> li)

void swap(list<T>& li)
{
	std::swap(_head, li._head);
	std::swap(_size, li._size);
}
list<T>& operator=(list<T> li)
{
	swap(li);
	return *this;
}

6、析构函数
~list();

void clear()
{
	iterator it = begin();
	while (it != end())
	{
		it = erase(it);
	}
	_size = 0;
}
~list()
{
	clear();
	delete _head;
	_head = nullptr;
}

三、迭代器

讲到list迭代器这里,我们先把迭代器类完善一下

template<class T,class Ref,class Ptr>//Ref表示引用,Ptr表示指针
struct ListIterator
{
    typedef ListNode<T> Node;
    typedef ListIterator<T,Ref,Ptr> Self;
    Node* _node;
    ListIterator(Node* node)
   			:_node(node)
    {}
    //*it
    Ref operator*()
    {
    	return _node->_data;
    }
    //it->
    Ptr operator->()
    {
    	return &_node->_data;
    }
    //++i
	Self& operator++()
	{
		_node = _node->_next;
		return *this;//返回的变量出了作用域不会被销毁用引用返回更合适
	}
	//i++
	Self operator++(int)
	{
		Self temp(*this);
		_node = _node->_next;
		return temp;//返回的变量出了作用域会被销毁用传值返回更合适
	}
	Self& operator--()
	{
		_node = _node->_prev;
		return *this;
	}
	Self operator--(int)
	{
		Self temp(*this);
		_node = _node->_prev;
		return temp;
	}
	Ref operator*()
	{
		return _node->_data;
	}
	bool operator!=(const Self& s)
	{
		return _node != s._node;//这里是用迭代器的地址去比较
	}
	bool operator==(const Self& s)
	{
		return _node == s._node;
	}
};

这个类里面的接口,我们重点讲解一下这个Ptr operator->();如下图:
在这里插入图片描述

iterator begin();
iterator end();
const_iterator begin();
const_iterator end();

template<class T>
class list
{
    typedef ListNode<T> Node;
    Node* _head;//哨兵位的头节点
    size_t _size;//链表数据个数
public:
    typedef ListIterator<T, T&, T*> iterator;
    typedef ListIterator<T, const T&, const T&> const_iterator;
    iterator begin()
    {
    	//return iterator(_head->_next);匿名对象
		return _head->_next;//单参数的构造函数隐式类型转换
	}
    iterator end()
    {
		return _head;
	}
    const_iterator begin()
    {
		return _head->_next;
	}
    const_iterator end()
    {
		return _head;
	}
}

四、数据的访问

1、front 访问头结点
T& front();

T& front()
{
	return _head->_next->_data;
}

const T& front()const;

const T& front() const
{
	return _head->_next->_data;
}

2、back 访问尾节点
T& back();

T& back()
{
	return _head->_prev->_data;
}

const T& back()const;

const T& back() const
{
	return _head->_prev->_data;
}

五、容量相关的函数

size 有效数据个数
size_t size()const;

size_t size() const 
{
	return _size;
}

empty 判断是否为空
bool empty()const;

bool empty()
{
   return _size == 0;
}

六、关于数据的增删查改操作

push_back 尾插数据
void push_back(const T& val) ;

void push_back(const T& val)
{
	Node* newnode = new Node(val);
	Node* tail = _head->_prev;

	tail->_next = newnode;
	newnode->_prev = tail;
	newnode->_next = _head;
	_head->_prev = newnode;

	_size++;
}

pop_back 尾删数据
void pop_back() ;

void pop_back()
{
	Node* tail = _head->_prev;
	Node* prev = tail->_prev;
	delete tail;
	prev->_next = _head;
	_head->_prev = prev;
	_size--;
}

push_front 头插数据
void push_front(const T& val);

void push_front(const T& val)
{
	Node* newnode = new Node(val);
	Node* next = _head->_next;
	//_head newnode next
	newnode->_next = next;
	next->_prev = newnode;
	newnode->_prev = _head;
	_head->_next = newnode;
	_size++;
}

pop_front 头删数据
pop_front();

void pop_front()
{
	Node* del = _head->_next;
	Node* next = del->_next;
	delete del;
	_head->_next = next;
	next->_prev = _head;
	_size--;
}

insert 在pos位置前插入值为val的节点
iterator insert(iterator pos, const T& val);

void  insert(iterator pos, const T& val)
{
	Node* cur = pos._node;
	Node* prev = cur->_prev;
	Node* newnode = new Node(val);
	//prev newnode cur
	prev->_next = newnode;
	newnode->_prev = prev;
	newnode->_next = cur;
	cur->_prev = newnode;

	_size++;
	
}

erase 删除pos位置的节点,返回该节点的下一个位置
iterator erase(iterator pos);

iterator erase(iterator pos)
{
	Node* cur = pos._node;
	Node* prev = cur->_prev;
	Node* next = cur->_next;
	delete cur;
	prev->_next = next;
	next->_prev = prev;
	_size--;
	return next;
}

clear 清除结点
void clear();

void clear()
{
	iterator it = begin();
	while (it != end())
	{
		it = erase(it);
	}
	_size = 0;
}

swap 交换两个链表
void swap(list<T>& l);

void swap(list<T>& li)
{
	std::swap(_head, li._head);
	std::swap(_size, li._size);
}

list篇到这里就结束了🎉,欢迎大家来指教我的下一篇stack和queue

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

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

相关文章

2024中国(重庆)人工智能展览会8月举办

2024中国(重庆)人工智能展览会8月举办 邀请函 主办单位&#xff1a; 中国航空学会 重庆市南岸区人民政府 招商执行单位&#xff1a; 重庆港华展览有限公司 【报名I59交易会 2351交易会 9466】 展会背景&#xff1a; 2024中国航空科普大会暨第八届全国青少年无人机大赛在…

linux Nginx安装与启动

一、先到官网下载Nginx 官网地址&#xff1a; http://nginx.org/en/download.html 我下载的是nginx-1.20.2 二、下载好的文件上传到服务器&#xff0c;然后解压 1、上传到指定的服务器地址&#xff0c;我这里是公司服务器&#xff0c;目录都是定义好的&#xff0c;自己玩建…

分层存储无法拯救 Kafka

01 引言 Apache Kafka 自诞生之日起&#xff0c;就以其卓越的设计和强大的功能&#xff0c;成为了流处理领域的标杆。它不仅定义了现代流处理架构&#xff0c;更以其独特的分布式日志抽象&#xff0c;为实时数据流的处理和分析提供了前所未有的能力。Kafka 的成功&#xff0…

element ui输入框后面带输入的字符数量

使用el-input的属性&#xff1a; maxlength&#xff1a;最长字符限制&#xff1b; show-word-limit&#xff1a;显示输入字符数量&#xff1b; 例&#xff1a; <el-input v-model"title" placeholder"请输入名称" maxlength"200" show-wor…

阮怀俊谈如何盘活和挖掘乡村文旅资源

近年来&#xff0c;浙江凭借高水平建设新时代美丽乡村&#xff0c;各项工作持续走在全国前列&#xff0c;最近&#xff0c;在国家发展改革委关于恢复和扩大消费措施的通知中也提到&#xff1a; “推广浙江‘千万工程’经验&#xff0c;建设宜居宜业和美乡村。实施文化产业赋能乡…

vue3专栏项目 -- 四、前后端结合(上)

一、前后端分离是什么 前面我们一直在和静态数据打交道&#xff0c;虽然流程可以跑个半通&#xff0c;但是静态数据还是给我们造成了诸多不便&#xff0c;现在我们是时候用上后端了。 现在的应用开发模式&#xff0c;自从SPA出现以后&#xff0c;前端和后端可以平行的进行对应…

智慧法治:AI技术如何赋能法律行业创新

&#x1f9d1; 作者简介&#xff1a;阿里巴巴嵌入式技术专家&#xff0c;深耕嵌入式人工智能领域&#xff0c;具备多年的嵌入式硬件产品研发管理经验。 &#x1f4d2; 博客介绍&#xff1a;分享嵌入式开发领域的相关知识、经验、思考和感悟&#xff0c;欢迎关注。提供嵌入式方向…

AI绘画动漫转真人详细教程

从小到大&#xff0c;我们看过的动漫、玩过的游戏有很多很多 但我们会发现里面的角色或者人物都是二次元的 我就会好奇这些动漫人物在现实中会长什么样 而现在&#xff0c;我们通过AI绘画竟然就能还原出来他们现实中的样子 除了动漫角色和游戏人物&#xff0c;古代的画像、经典…

【笔记】从零开始做一个男性人体的流程/躯干篇(超级详细)

躯干整体 大体 1.创建一个正方体&#xff0c;摆好位置 2.实例呀啥的都搞好 3.胸部它是一个前窄后宽的结构 斜方肌 臀部 1.臀部是前宽后窄的结构 2.我们再去侧面调整以下 胸椎向上倾斜&#xff0c;盆骨向下倾斜。脊椎是s形的 3.真实的身体没有这么方正&#xff0c;所以微调…

3kCTF2021 echo klibrary

文章目录 前言echoklibrary 前言 今天状态不好&#xff0c;很多事情都不想干&#xff0c;就做一做简单的题目 echo 内核版本&#xff1a;v5.9.10smap/smep/kaslr 开启modprobe_path 可写 题目给了源码&#xff0c;非常简单就是无限次的任意地址读写&#xff1a; #include …

Lombok注解详解

文章目录 注解详解lombok包下注解汇总- Getter- Setter- ToString- EqualsAndHashCode- Data- Value- NonNull- NoArgsConstructor- AllArgsConstructor- RequiredArgsConstructor- Builder- Synchronized- Cleanup- Singular- Generated- SneakyThrows- val- var experimental…

pwn(安装环境)

从安装虚拟机开始 1.先安装vmware 详细教程&#xff1a; VMware下载安装教程(超详细)-CSDN博客 2.然后下载虚拟机镜像文件 进入下面这个链接下载 Get Kali | Kali LinuxHome of Kali Linux, an Advanced Penetration Testing Linux distribution used for Penetration Te…

Linux安装配置CGAL,OpenCV和Gurobi记录

安装Qt&#xff0c;查看当前的Qt版本&#xff0c;需要至少满足v5.12 qmake -v安装CGAL&#xff0c;The Computational Geometry Algorithms Library (cgal.org) CGAL v5.6.1&#xff1a;https://github.com/CGAL/cgal/releases/download/v5.6.1/CGAL-5.6.1.tar.xz 确保C编译…

5款可用于LLMs的爬虫工具/方案

5款可用于LLMs的爬虫工具/方案 Crawl4AI 功能: 提取语义标记的数据块为JSON格式&#xff0c;提供干净的HTML和Markdown文件。 用途: 适用于RAG&#xff08;检索增强生成&#xff09;、微调以及AI聊天机器人的开发。 特点: 高效数据提取&#xff0c;支持LLM格式&#xff0c;多U…

改变浏览器大小,图片(img)内容居中显示img标签,不是背景图

改变浏览器大小,图片&#xff08;img&#xff09;内容居中显示&#xff0c;img标签&#xff0c;不是背景图 效果直接上图&#xff1a; 上代码&#xff1a; <!DOCTYPE html> <html> <head><title>测试图片居中显示&#xff0c;高度不变只变宽度<…

HCIP【BGP综合实验】

目录 一、实验拓扑图&#xff1a; 二、实验要求&#xff1a; 三、实验思路&#xff1a; 四、实验步骤&#xff1a; 1、进行网段的子网划分&#xff08;整个实验总共有19条网段&#xff09;&#xff1a; (1)首先&#xff0c;根据实验要求&#xff0c;将172.16.0.0/16全部划…

C语言学习(九)多文件编程 存储类型 结构体

目录 一、多文件编程&#xff08;一&#xff09;不写头文件的方方式进行多文件编程 &#xff08;二&#xff09;通过头文件方式进行多文件编程&#xff08;1&#xff09;方法&#xff08;2&#xff09;头文件守卫 &#xff08;三&#xff09; 使用多文件编程实现 - * / 功能 二…

系统设计中的泛化调用

背景 目前在学习一些中间件&#xff0c;里面看到了一个词是叫泛化调用&#xff0c; 其实这个场景在JAVA中比较常见。我们常用的有反射&#xff0c;反射就是我知道类名称、类方法和参数&#xff0c;调用一个Object的类&#xff0c;但是在HTTP或者RPC远程调用过程中&#xff0c;…

服务异步通讯MQ

同步调用存在的问题: 异步调用方案: RabbitMQ安装: 第一种:在线拉取 docker pull rabbitmq:3-management 第二种:将已有的安装包放入再用load加载 我这里放到tmp包里边 然后:cd /tmp docker load -i mq.tar 加载进去 然后运行mq容器 docker run \-e RABBITMQ_DEFAULT_USER…

【一步一步了解Java系列】:了解Java与C语言的运算符的“大同小异”

看到这句话的时候证明&#xff1a;此刻你我都在努力~ 加油陌生人~ 个人主页&#xff1a; Gu Gu Study ​​ 专栏&#xff1a;一步一步了解Java 喜欢的一句话&#xff1a; 常常会回顾努力的自己&#xff0c;所以要为自己的努…