c++链表(list)

前言

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

具体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.rhkb.cn/news/407659.html

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

相关文章

CUDA指南-CUDA编程基础

CUDA编程基础是开始利用GPU进行并行计算的起点。以下是一些入门步骤和概念&#xff1a; Hello World&#xff1a;第一个CUDA程序 编写CUDA核函数&#xff1a; 创建一个简单的核函数&#xff0c;例如一个向量加法操作。核函数用 global 修饰&#xff0c;表示它将在GPU上执行。…

React 学习——useMemo

useMemo使用场景&#xff1a;消耗非常大的计算&#xff0c;例如递归 import { useMemo, useState } from react; // 缓存&#xff1a;消耗非常大的计算&#xff0c;例如递归 function fib(n){console.log(fib);if(n < 3)return 1;return fib(n-2) fib(n-1); }const App (…

Linux | 文件系统进阶:Inode与软硬链接艺术剖析

当时共我赏花人&#xff0c;点检如今无一半。 - 《木兰花》(晏殊) 2024.8.24 目录 1. 文件系统的基本概念 1.1 ls -l命令查看目录信息 1.2 stat命令查看具体文件的详细信息 1.3 inode ext2文件系统的主要组成部分&#xff1a; 例子&#xff1a;创建main.c文件 文件的创建步骤&a…

谷歌登录的时候,要求在手机的通知点是,并按数字来验证身份,但是手机通知栏没有收到通知和数字,原因是什么,怎么办?

前两天&#xff0c;有个朋友联系到GG账号服务&#xff0c;说他的一个谷歌账号在新设备登录的时候&#xff0c;提示说要在手机的通知栏点击谷歌发来的通知&#xff0c;点击是确认&#xff0c;并且要点按相应的数字。 但问题是他反复刷新手机的通知栏都没有看到谷歌发来的通知&a…

xml打印模板解析-SAAS本地化及未来之窗行业应用跨平台架构

一、为何要自己设置打印模板系统 1.确保自由知识产权 2.支持跨平台&#xff1a;物联网&#xff0c;自助终端&#xff0c;电脑&#xff0c;web&#xff0c;C#&#xff0c;jsp,android,java,php 等多种语言 二、xml 代码解析 package CyberWinPHP.Cyber_Plus;import java.io.…

区块链浪潮:Web3时代的数字经济新格局

随着科技的迅猛发展&#xff0c;全球经济正迎来一场前所未有的变革&#xff0c;区块链技术正在其中扮演着关键角色。Web3作为下一代互联网的核心&#xff0c;正在通过区块链技术重塑数字经济的格局&#xff0c;为全球市场带来新的机遇和挑战。这场以去中心化为特征的技术革命&a…

进阶岛 - MindSearch(CPU版)部署到github codespace

实践结论写在前面&#xff1a; github codespace很好用&#xff0c;丝滑、快速huggingface的space管理代码也很好用&#xff0c;自动编译、容器打包、发布一气呵成mindsearch很好用&#xff0c;效果超出预想。planner和searcher的交互丝滑、有效率。mindsearch内置搜索可解决很…

JAVA基础面试题总结(十四)——JVM(下)

类文件结构详解 什么是字节码&#xff1f; 在 Java 中&#xff0c;JVM 可以理解的代码就叫做字节码&#xff08;即扩展名为 .class 的文件&#xff09;&#xff0c;它不面向任何特定的处理器&#xff0c;只面向虚拟机。Java 语言通过字节码的方式&#xff0c;在一定程度上解决…

45.5【C语言】typedef

目录&#xff1a; *全称 *格式 一般指针 数组指针 函数指针 *细节 *全称 type define 类型&#xff08;重新&#xff09;定义&#xff08;或命名&#xff09;&#xff0c;可简化输入 *格式 1.非指针类型: typedef 类型 简化名称 typedef signed long long k; signed long …

WPF中的可视化树(VisualTree)和逻辑树(LogicalTree)

可视化树和逻辑树 我们先来理解一下什么是可视化树和逻辑树。 可视化树&#xff1a;包含最初指定的大多数元素&#xff08;在XAML或.cs中&#xff09;以及控件模板中的元素。 通俗点来讲&#xff0c;就是整个元素的构成树&#xff0c;从最上面的结点到最后一个结点&#xff…

SpringCache源码解析(一)

一、springCache如何实现自动装配 SpringBoot 确实是通过 spring.factories 文件实现自动配置的。Spring Cache 也是遵循这一机制来实现自动装配的。 具体来说,Spring Cache 的自动装配是通过 org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration 这个类来…

Maven的使用

Maven 是一个项目管理工具&#xff0c;它基于项目对象模型&#xff08;POM&#xff0c;Project Object Model&#xff09;的概念&#xff0c;通过一小段描述信息&#xff08;pom.xml&#xff09;来管理项目的构建、报告和文档。Maven 提供了一个标准化的方式来构建项目&#xf…

MyBatis-Plus 三、(进阶使用)

一、typeHandler 的使用 1、存储json格式字段 如果字段需要存储为json格式&#xff0c;可以使用JacksonTypeHandler处理器。使用方式非常简单&#xff0c;如下所示&#xff1a; 只需要加上两个注解即可&#xff1a; TableName(autoResultMap true) 表示自动…

第五节:Nodify 节点位置设置

引言 如果你尝试过前几节的代码&#xff0c;会发现节点都是出现在0,0 位置&#xff0c;及编辑器左上角。编辑器作为最外层的交互控件&#xff0c;内部封装了节点容器ItemContrainer&#xff0c;我们通过样式属性对Loaction做绑定。本节将介绍如何配置节点位置。 1、节点位置 …

DHCP DNS 欺骗武器化——实用指南

DHCP 枚举 在我们之前的文章中,我们分享了 DHCP DNS 欺骗背后的理论。实际上,需要几条信息才能有效地执行我们描述的攻击。对于攻击者来说幸运的是,发现DHCP 服务器并了解其配置的能力是 DHCP 协议的一部分,这使得侦察过程变得微不足道。 在以下章节中,我们将描述攻击者…

Git的使用教程及常用语法02

四.将文件添加到仓库 创建仓库 git init查看仓库的状态 git status 添加到暂存区 git add提交 git commitgit status 可以查看当前仓库的状态信息&#xff0c;例如包含哪些分支&#xff0c;有哪些文件以及这些文件当前处在怎样的一个状态。 由于当前没有存储任何的东西&…

基于Python的机器学习系列(7):多元逻辑回归

在本篇博文中&#xff0c;我们将探讨多元逻辑回归&#xff0c;它是一种扩展的逻辑回归方法&#xff0c;适用于分类数量超过两个的场景。与二元逻辑回归不同&#xff0c;多元逻辑回归使用Softmax函数将多个类别的概率输出映射到[0, 1]范围内&#xff0c;并确保所有类别的概率和为…

PMBOK® 第六版 控制范围

目录 读后感—PMBOK第六版 目录 结果固然重要&#xff0c;过程同样不可或缺。过程不仅是通往预期成果的途径&#xff0c;也是个人和团队能力提升与经验积累的关键阶段。过程中的每一步都是学习和成长的机会&#xff0c;每一次尝试都能激发创新&#xff0c;而公正透明的流程更增…

TCP BBR 数学模型完整版

今天顺带加入了 bbr 的所有状态和所有流程&#xff0c;获得以下的方程组&#xff1a; C Bltbw&#xff0c;R RtProp&#xff0c;T_r ProbeRTT 周期&#xff0c;g1 Startup gain&#xff0c;g2 ProbeBW gain。设 x estimated bandwidth&#xff0c;r round trip time&am…

MySQL 数据库深度解析:安装、语法与高级查询实战

一、引言 在现代软件开发和数据管理领域中&#xff0c;MySQL 数据库凭借其高效性、稳定性、开源性以及广泛的适用性&#xff0c;成为了众多开发者和企业的首选。无论是小型项目还是大型企业级应用&#xff0c;MySQL 都能提供可靠的数据存储和管理解决方案。本文将深入探讨 MyS…