c++链表(list)

news2024/12/28 10:43:29

前言

链表作为一个常见的数据结构,在高频插入删除的场景下有独特的优势,在内存的使用上也极少有浪费可以按需申请。今天我们就来简单的学习一下这种数据结构,链表也有很多不同的实现,我们这里和标准库保持一致,实现带头双线循环链表

具体list类的描述可以参考list - C++ Reference (cplusplus.com)

在不同的编译器下string类的实现各有差异,这里我们使用的是Microsoft Visual Studio Community 2022 (64 位) - Current 版本 17.8.5

链表的结构

在正式学习链表之前,我们先来简单的认识一下带头双向循环链表的结构。

带头指的是有哨兵位的头节点,哨兵位指的是一个没有任何数据的节点,作用是标识首位节点(如果是单向指标识头节点),因为有哨兵位的存在,很多操作得以简化

双向指的是每个节点都有标识上一个节点和下一个节点的地址

循环指的是尾节点不在指向nullptr而是指向哨兵位

首先,我们需要实现一个节点的类,用来描述节点的属性

template<class T>
	class listNode
	{
	public:
		//listNode() {}
		listNode(const T& a=T())
		{
			data = a;
		}

		//private:
		listNode* next = nullptr;
		listNode* previous = nullptr;
		T data = T();
	};

 这里我们还是使用类模板,以适应存储各种类型的数据

全部属性用public描述,因为这个类在后面要经常被使用到,用struct可以实现一样的效果,用友元类也可以使用private描述,这里就简化这些操作了

此时如果我们需要再链表中使用节点,需要先声明节点的属性

typedef listNode<T> Node;

现在我们就可以来搭建一个链表类的框架出来

#pragma once
#include <iostream>
#include <algorithm>
//#include <assert.h>

using namespace std;

namespace zzzyh
{
    template<class T>
	class list
	{
public:
private:
		Node* head;
		size_t sz = 0;
	};
}

其中head表示哨兵位的地址,size用来标识有效元素的个数

 构造函数

allocator是空间配置器,这里我们还是先忽略这个点

第一个是默认构造函数

list()
		{
			empty_init();
		}

这里empty_init是初始化哨兵位,因为经常用到我们可以封装为一个函数

void empty_init()
		{
			head = new Node;
			head->next = head;
			head->previous = head;
			sz = 0;
		}

第二个是使用n个默认值进行初始化

第三个是使用迭代器区间进行初始化

第四个是拷贝构造

list(const list&s)
		{
			empty_init();
			for (auto &x : s)
			{
				push_back(x);
			}
		}

顺带,我们将赋值重载也实现

		list<T>& operator=(const list&s)
		{
			list<T> rem(s);
			swap(rem);
			return *this;
		}

这里的push_back功能是尾插一个元素,我们先使用在后面再实现

这里swap可以实现两个链表的交换,还是先使用在后面我们再来实现

这里还是会有深浅拷贝的问题,只有不能两个节点指向同一块空间,不能多次释放一块空间

在最后我们介绍一个c++11引入的构造方式

这里initializer_list是一个类

这个类我们不在这里不展开只介绍使用

#define _CRT_SECURE_NO_WARNINGS 1
#include "list.h"
#include <list>

using namespace std;

int main()
{
	list<int> list1 = { 1,2,3,4,5 };
	list<int> list2({ 6,7,8,9,0 });
	for (int i : list1)
	{
		cout << i << " ";
	}
	cout << endl;
	for (int i : list2)
	{
		cout << i << " ";
	}
	cout << endl;
	return 0;
}

下面我们来实现一下这种功能

list(initializer_list<T> li)
		{
			empty_init();
			for (auto& a : li)
			{
				push_back(a);
			}
		}

这个构造方式不仅仅再list中适用,所有实现这个接口的容器均可以使用,具体哪些容器实现类还需要再查询文档

析构函数

析构函数可以先将使用有效的节点释放(clear)再释放哨兵位头节点

~list()
		{
			clear();
			delete head;
		}

我们再来实现一下clear

		void clear()
		{
			iterator beg = begin();
			while (beg != end())
			{
				beg = erase(beg);
			}
		}

这里的erase可以删除指定迭代器指向的位置,这里还是先使用在后面会实现其功能

迭代器

链表的迭代器相比于前面我们介绍的顺序表要复杂很多,因为链表的内存空间是不连续的,这就意味着++这种操作符需要重载为新的含义

我们这里可以把迭代器实行为一个类来描述,因为只有类才能实现运算符重载

template<class T>
	class list_iterator
	{
	public:
		typedef listNode<T> Node;
		Node* cut;
		list_iterator(Node* c)
			:cut(c)
		{}
		T& operator*() const
		{
			return cut->data;
		}

		T* operator->() const
		{
			return &(cut->data);
		}

		list_iterator operator++()
		{
			cut = cut->next;
			return *this;
		}
		list_iterator operator--() {
			cut = cut->previous;
			return *this;
		}

		list_iterator operator++(int)
		{
			cut = cut->next;
			return cut->previous;
		}
		list_iterator operator--(int) {
			cut = cut->previous;
			return cut->previous;
		}
		list_iterator operator+(size_t i) {
			Node* ret = cut;
			while (i != 0)
			{
				ret = ret->next;
				i--;
			}
			return ret;
		}
		list_iterator operator-(size_t i) {
			Node* ret = cut;
			while (i != 0)
			{
				ret = ret->previous;
				i--;
			}
			return ret;
		}

		bool operator!=(list_iterator pos) const
		{
			return cut != pos.cut;
		}

		bool operator==(list_iterator pos) const
		{
			return cut == pos.cut;
		}
	};

同样我们可以实现const迭代器

template<class T>
	class list_const_iterator
	{
	public:
		typedef listNode<T> Node;
		Node* cut;
		list_const_iterator(Node* c)
			:cut(c)
		{}
		const T& operator*() const
		{
			return cut->data;
		}

		const T* operator->() const
		{
			return &(cut->data);
		}

		list_const_iterator operator++()
		{
			cut = cut->next;
			return *this;
		}
		list_const_iterator operator--() {
			cut = cut->previous;
			return *this;
		}

		list_const_iterator operator++(int)
		{
			cut = cut->next;
			return cut->previous;
		}
		list_const_iterator operator--(int) {
			cut = cut->previous;
			return cut->previous;
		}
		list_const_iterator operator+(size_t i) {
			Node* ret = cut;
			while (i != 0)
			{
				ret = ret->next;
				i--;
			}
			return list_iterator(ret);
		}
		list_const_iterator operator-(size_t i) {
			Node* ret = cut;
			while (i != 0)
			{
				ret = ret->previous;
				i--;
			}
			return list_iterator(ret);
		}

		bool operator!=(list_const_iterator pos) const
		{
			return cut != pos.cut;
		}

		bool operator==(list_const_iterator pos) const
		{
			return cut == pos.cut;
		}
	};

此时我们发现,这两个类高度相似,我们也可以用到类模板的思想实现

template<class T,class Ref,class Ptr>
	class list_iterator
	{
	public:
		typedef listNode<T> Node;

		Node* cut;
		list_iterator(Node* c)
			:cut(c)
		{}
		Ref operator*()
		{
			return cut->data;
		}

		Ptr operator->()
		{
			return &(cut->data)	;
		}

		list_iterator operator++()
		{
			cut = cut->next;
			return *this;
		}
		list_iterator operator--() {
			cut = cut->previous;
			return *this;
		}

		list_iterator operator++(int)
		{
			cut = cut->next;
			return cut->previous;
		}
		list_iterator operator--(int) {
			cut = cut->previous;
			return cut->previous;
		}
		list_iterator operator+(size_t i) {
			Node* ret = cut;
			while (i != 0)
			{
				ret = ret->next;
				i--;
			}
			return list_iterator(ret);
		}
		list_iterator operator-(size_t i) {
			Node* ret = cut;
			while (i != 0)
			{
				ret = ret->previous;
				i--;
			}
			return list_iterator(ret);
		}

		bool operator!=(list_iterator pos) const
		{
			return cut != pos.cut;
		}

		bool operator==(list_iterator pos) const 
		{
			return cut == pos.cut;
		}
	};

如果此时我们需要再链表里使用迭代器,同样需要先声明迭代器的属性

typedef list_iterator<T,T&,T*> iterator;
typedef list_iterator<T, const T&, const T*> const_iterator;

同理如果上面那两种未使用类模板实现的迭代器在使用时也需要先声明再使用

typedef list_iterator<T> iterator;
typedef list_const_iterator<T> const_iterator;

此时我们可以实现begin和end了

iterator begin()
		{
			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);
		}

其他的获得迭代器的函数我们就不再介绍了,和前面介绍过的容器功能相同

 容量相关

size(),可以获得有效节点的个数,不包含哨兵位头节点

size_t size()
		{
			return sz;
		}

empty(),判断此链表是否为空(即没有有效元素,还是不包含哨兵位头节点)

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

访问(查)相关

front(),可以返回首节点的值的引用

back(),可以返回最后一个节点的值的引用

增删改相关

push_back(),可以尾插一个节点

void push_back(const T& a)
		{
			Node* newNode = new Node(a);
			if (head->next == head) 
			{
				newNode->next = newNode->previous = head;
				head->next = head->previous = newNode;
			}
			else
			{
				newNode->next =  head;
				newNode->previous = head->previous;
				head->previous->next = newNode;
				head->previous = newNode;
			}
			sz++;
			//insert(end(), a);
		}

insert(),可以在指定迭代器之前的位置插入一个节点,迭代器不失效

void insert(iterator pos, const T& data)
		{
			if (pos == end())
			{
				push_back(data);
				return;
			}
			else {
				Node* newNode = new Node(data);
				if (pos == begin()) {
					newNode->previous = head;
					newNode->next = head->next;
					head->next->previous = newNode;
					head->next = newNode;
				}
				else {
					newNode->previous = pos.cut->previous;
					newNode->next = pos.cut;
					pos.cut->previous->next = newNode;
					pos.cut->previous = newNode;
				}
			}
			sz++;
		}

push_front(),可以头插一个节点

void push_front(const T& a)
		{
			insert(begin(), a);
		}

erase(),可以删除指定迭代器的节点,并且返回当前迭代器的下一个位置

iterator erase(iterator pos)
		{
			if (pos.cut == head) {
				return pos;
			}
			Node* left = pos.cut->previous;
			Node* right = pos.cut->next;
			left->next = right;
			right->previous = left;
			delete pos.cut;
			sz--;
			return right;
		}

pop_back(),可以尾删尾节点

void pop_back()
		{
			erase(--end());
		}

pop_front(),可以头删头节点

void pop_front()
		{
			erase(begin());
		}

swap(),可以交换两个链表的内容,这里单独实现是为了提高交换效率

void swap(list& rem)
		{
			swap(sz,rem.sz);
			swap(head, rem.head);
		}

迭代器失效

这里还是就有迭代器失效的问题,但只要在删除时会失效传入的迭代器,其他迭代器不受影响。标准库和我们这的结局方案相同,在调用删除时会将新的迭代器(即删除元素的下一个位置)返回,及时更新使用即可

结语

以上便是今天的全部内容。如果有帮助到你,请给我一个免费的赞。

因为这对我很重要。

编程世界的小比特,希望与大家一起无限进步。

感谢阅读!

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

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

相关文章

UDP通信函数补充 | TCP

UDP流程补充&#xff1a; recvfrom() 这是一个系统调用&#xff0c;用于从套接字接收数据的函数。该函数通常与无连接的数据报服务&#xff08;如 UDP&#xff09;一起使用&#xff0c;但也可以与其他类型的套接字使用。 函数原型为&#xff1a; ssize_t recvfrom(int sock…

使用Node-RED实现和部署物联网入侵检测的机器学习管道

整理自 《Implementing and Deploying an ML Pipeline for IoT Intrusion Detection with Node-RED》&#xff0c;由 Yimin Zhang 等人撰写&#xff0c;发表于 2023 年 CPS-IoT Week Workshops。以下是根据提供的 PDF 内容整理的论文的详细主要内容&#xff1a; 摘要 (Abstra…

Linux入门——09 共享内存

1.共享内存原理 OS内的每个进程都会有自己的内核结构&#xff08;task_struct&#xff09;和虚拟地址空间,通过页表与物理内存进程映射。 如果让两个不同的进程共享内存&#xff0c;首先就是在内存中申请一块空间&#xff08;共享内存&#xff09;&#xff0c; 然后将建立好…

Unity XR Interaction Toolkit 踩坑记录

1&#xff1a;按下 grap/select 键 物品直接飞到手上 2 按下 grap/select 键 物品一点点的想自己移动

《机器学习》—— AUC评估指标

文章目录 一、什么是AUC&#xff1f;1、什么是ROC曲线&#xff1f;2、ROC曲线的绘制 二、如何计算AUC的值三、代码实现AUC值的计算四、AUC的优缺点 一、什么是AUC&#xff1f; 机器学习中的AUC&#xff08;Area Under the Curve&#xff09;是一个重要的评估指标&#xff0c;特…

走进虚拟机逃逸技术之VMware Escape漏洞CVE-2023-20872复现

走进虚拟机逃逸技术之VMware Escape漏洞CVE-2023-20872复现 技术分享 技术分享 起初&#xff0c;为了学习虚拟机逃逸相关技术&#xff0c;也为了搞懂硬件虚拟化。于是请教了某巨佬后告诉我一本书&#xff0c;看完之后为了验证我理解到的硬件虚拟化及虚拟化逃逸原理是否正确&am…

图书管理系统详细设计

需求概述 按照需求分析文档中的规格要求&#xff0c;使用条形码扫描器进书、借书、还书&#xff0c;使得信息传递准确、流畅。同时&#xff0c;系统最大限度地实现易安装&#xff0c;易维护性&#xff0c;易操作性&#xff0c;运行稳定&#xff0c;安全可靠。 软件结构 系统由…

如何让虚拟机识别到宿主机的USB设备

我的实验环境&#xff1a; Windows宿主机VirtualBox虚拟化软件一个Linux虚机一个8G的USB磁盘 首先要让虚拟机能看到宿主机的USB设备&#xff0c;这是在VirtualBox中设置的。 选中虚机&#xff0c;右键选择“设置”菜单&#xff0c;再单击“USB设备”&#xff1a; 选中“启用…

Python | Leetcode Python题解之第365题水壶问题

题目&#xff1a; 题解&#xff1a; class Solution:def canMeasureWater(self, x: int, y: int, z: int) -> bool:if x y < z:return Falseif x 0 or y 0:return z 0 or x y zreturn z % math.gcd(x, y) 0

Alembic:python中数据库迁移的瑞士军刀

Alembic 简介 Alembic 是由 SQLAlchemy 的创始人 Mike Bayer 设计的一个数据库迁移工具。它不仅支持自动迁移脚本生成&#xff0c;还允许开发者手动编辑迁移脚本来满足特定的需求。Alembic 通过提供一个环境来跟踪数据库模式的变更历史&#xff0c;确保数据库的版本与应用代码…

推荐一个完全自由的目录设计网站

引言 如果我们能通过网站出一本书&#xff0c;这将是一件很酷的事。 事实上&#xff0c;我们通过网站发布知识&#xff0c;最常见的是写博客。 这二者有什么区别呢&#xff1f; 书本的知识内容有很强的逻辑性、系统性。而博客是随心所欲的&#xff0c;一时灵感来了就写一篇…

关闭Chrome快捷键

chrome是没办法改变快捷键以及屏蔽快捷键的&#xff0c;需要安装插件&#xff1a;shortkey 保证插件是开启的 不用做其他设置所有快捷键已被关闭

OAPT:用于双JPEG伪影去除的偏移感知分区的Transformer

OAPT: Offset-Aware Partition Transformer for Double JPEG Artifacts Removal https://github.com/QMoQ/OAPT 2408.11480 (arxiv.org) 基于深度学习的方法在去除单个JPEG伪影任务中表现出了显著的性能。然而&#xff0c;现有方法在处理双重JPEG图像时往往会退化&#xff0c…

127-隧道搭建穿透上线FRPNPSNgrok

使用了几种工具将会一一介绍 ngrokru 项目地址&#xff1a;Sunny-Ngrok内网转发内网穿透 - 国内内网映射服务器 这个网站现在要实名认证&#xff08;还得花2元解锁&#xff09; 用这种在线的网站怎么说呢&#xff0c;真不如自己买个云服务器用下面的frp&#xff0c;毕竟流量…

Python3:多行文本内容转换为标准的cURL请求参数值

背景 在最近的工作中&#xff0c;经常需要处理一些接口请求的参数&#xff0c;参数来源形式很多&#xff0c;可能是Excel、知识库文档等&#xff0c;有些数据形式比较复杂&#xff0c;比如多行或者包含很多不同的字符&#xff0c;示例如下&#xff1a; **客服质检分析指引** …

多个程序监听不同网卡的相同端口、相同网卡不同IP的相同端口

1 概述 一个主机上的多个程序监听同一个端口&#xff0c;是否一定存在冲突&#xff1f;如果是多网卡、单网卡多IP的情景下&#xff0c;多个程序是可以独立监听的。 2 多个程序监听不同网卡的相同端口 3 多个程序监听同一个网卡不同IP的相同端口 4 小结 多个程序监听同一个网…

生成式人工智能会导致人工智能崩溃吗

况可能很快就会发生变化。 从定义上讲&#xff0c;LLM 需要大量数据&#xff0c;而且所使用的数据集越来越大。根据缩放定律[2]&#xff0c;要提高性能&#xff0c;必须同时增加参数数量和训练标记数量&#xff08;后者被认为是最重要的因素&#xff09;。 这些数据集包含人类产…

0x03 ShowDoc 文件上传漏洞(CNVD-2020-26585)复现

参考&#xff1a;ShowDoc文件上传漏洞&#xff08;CNVD-2020-26585&#xff09;_showdoc漏洞-CSDN博客 一、fofa 搜索使用该工具的网站 网络空间测绘&#xff0c;网络空间安全搜索引擎&#xff0c;网络空间搜索引擎&#xff0c;安全态势感知 - FOFA网络空间测绘系统 "S…

ZMQ请求应答模型

案例一 这个案例的出处是ZMQ的官网。请求段发送Hello&#xff0c;应答端回复World。 ZMQ Request(client) #include <string> #include <iostream> #include <zmq.hpp>using namespace std; using namespace zmq; // 使用 zmq 命名空间int main() {// ini…

知识竞赛答题设备及答题方式有哪些

根据我们多年的知识竞赛承办经验&#xff0c;我来谈谈在知识竞赛中常用的答题设备和答题方式。 一、常用答题设备 1.电脑 如果电脑资源充足&#xff0c;可以用笔记本电脑进行答题&#xff0c;笔记本电脑可以采取有线或无线方式进行连网&#xff0c;可以根据情况选择连网方案&…