【数据结构】链表:看我如何顺藤摸瓜

news2024/11/26 18:26:51

在这里插入图片描述

  • 👑专栏内容:数据结构
  • ⛪个人主页:子夜的星的主页
  • 💕座右铭:日拱一卒,功不唐捐

文章目录

  • 一、前言
  • 二、链表
    • 1、定义
    • 2、单链表
      • Ⅰ、新建一个节点
      • Ⅱ、内存泄漏
      • Ⅲ、插入一个节点
      • Ⅳ、销毁所有节点
      • Ⅴ、反转一个链表
    • 3、双向链表
    • 4、循环链表
      • Ⅰ、单向循环链表
      • Ⅱ、双向循环链表
      • Ⅲ、循环链表总结
      • Ⅳ、一些OJ题
        • ①、环形链表
        • ②、快乐数
  • 三、总结
    • 1、区别
    • 2、优点
    • 3、缺点


一、前言

前面介绍了线性结构中的顺序表,顺序表的随机访问速度非常快,但是它最大的缺点就是插入和删除的时候要移动大量元素。而链表这一数据结构就能完美的解决这一问题。那么链表是如何解决的呢?

在这里插入图片描述

二、链表

1、定义

链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素。链表中的数据是以结点来表示的,每个结点的构成:元素(数据元素的映象) + 指针(指示后继元素存储位置),元素就是存储数据的存储单元,指针就是连接每个结点的地址数据。

由于是分散存储,为了能够体现出数据元素之间的逻辑关系,每个数据元素在存储的同时,要配备一个指针,用于指向它的直接后继元素,即每一个数据元素都指向下一个数据元素(最后一个指向NULL(空))。
链表中每个元素本身由两部分组成:
● 数据域:存放数据。
● 指针域:存放指向后继结点的地址;
链表的第一个结点被称为:头节点

正因为如此,我们只需要记住链表中的头节点就行,顺着头节点这个藤,逐渐的可以找到链表中其他所有的节点。

在这里插入图片描述

typedef struct LinkNode {
		int data;//数据域
		struct LinkNode *next;//指针域
}Node;

在这里插入图片描述

2、单链表

单链表,顾名思义。链表指向的只有一个方向。

在这里插入图片描述

Ⅰ、新建一个节点

Node *getNewNode(int val) {
    Node *p = (Node *)malloc(sizeof(Node)); //(1)
    p->data = val;  	//(2)
    p->next = NULL;	    //(3)
    return p;			//(4)
}

(1)开辟一个空间用来存储新的节点
(2)将新建节点的数据放入数据域中
(3)将新节点的指针域置为 NULL
(4)返回新建的节点

Ⅱ、内存泄漏

在进行插入操作之前,先搞明白一个概念,那就是内存泄露

内存泄漏:由于疏忽或错误造成程序未能释放已经不再使用的内存。
在这里插入图片描述
就拿链表来看,我们知道的只有程序内部的头地址,依据头地址中存储的下一个链表的地址来找到下一个链表,从而,拔出萝卜带出泥,找到所有的链表。但是试想一下,加入我们在操作的过程中不小心把其中一个链表的地址弄丢了呢?
在这里插入图片描述
就像这样,因为错误的插入,导致弄丢了②和③的地址,导致②和③这两个链表数据在内存内部中无法被找到,也无法被清理。

在这里插入图片描述

Ⅲ、插入一个节点

如下图,我们想在①和②中间插入④,或者这样说,将④插入链表的2号位置,应该怎么操作呢?

在这里插入图片描述

首先当然应该让p指针走向2号位置的前一个位置,也就是1号位置处。

然后,很多人就会犯这样的错误

p->next = node; //(1)
node ->next = p->next ; //(2)

这样看似没有问题,但是仔细看一下,在执行完(1)后,p->next已经不是②的地址了。你再执行(2)其实是将node->next指向node自己。

在这里插入图片描述
这样就造成了前文说的内存泄漏,也就是说②和③从此就会一直呆在你的内存里面,你调用不了也销毁不了这两个链表。所以,我们插入结点时,一定要注意操作的顺序,要先将结点 node next 指针指向下一个节点地址,再把结点 ① 的 next指针指向结点node,这样才不会丢失指针,导致内存泄漏。所以,对于刚刚的插入代码,我们只需要把第 1 行和第 2 行代码的顺序颠倒一下就可以了。

Node *insert(Node *head,int pos ,int val) //(1)
{
	if(pos ==0) //(2)
	{
		Node *p = getNewNode (val);
		p->next = head;
		return p;
	}
	Node *p = head; 
	for(int i =1;i<pos;i++) 
		p = p->next;      //(3)
	Node *node = getNewNode(val); //(4)
	node ->next = p->next ;  //(5)
	p->next = node;  //(6)
	return head;
}

(1)pos是要插入的位置
(2)pos等于0就是放在头地址处
(3)让p指针走向待插入位置的前一个位置
(4)创建一个要插入的节点
(5)先让新建节点的指针指向待原本插入位置的节点地址
(6)让待前一个位置的指针指向新建要插入的节点

上面的代码自然能够很好的完成插入操作,但是这样代码写起来就会很繁琐,而且也容易出错。如何来更好的解决这个问题呢?仔细看下上面的插入操作,代码较长的原因,主要是怕插入的位置是头位置。所以我们要对这一情况进行特殊处理。但是,如果我们建立一个虚拟的头节点呢?这样不就可以解决找个问题了吗?
在这里插入图片描述

Node *insert(Node *head,int pos ,int val)
{
	Node new_head,*p = &new_head;  //(1)
	Node *node = getNewNode(val); 
	new_head.next = head //(2)
	for(int i=0;i<pos;i++) p= p->next; //(3)
	node->next = p->next; //(3)
	p->next = node;//(3)
	return new_head.next; //(4)
}

(1)新建一个虚拟的头节点
(2)让虚拟头节点的指针指向真实的头地址
(3)进行插入操作
(4)返回虚拟头节点的指向的地址,也就是真实的头地址

在这里插入图片描述

Ⅳ、销毁所有节点

注意:销毁所有的节点不能直接free(head),因为你如果直接释放了头节点,那么你根本找不到后面其他节点。
所以,应该新建一个指针,指向要销毁节点的下一个节点,再不断的更新这个指针指向的位置,依次销毁所有节点。切记,万万不可只销毁头节点。

void clear(Node *head) {
    if (head == NULL) return ;
    for (Node *p = head, *q; p; p = q) //(1)
    {
        q = p->next;	//(1)
        free(p);	    //(2)
    }
    return ;
}

(1)循环不断更新p指针指向的节点位置
(2)销毁节点

Ⅴ、反转一个链表

给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。

在这里插入图片描述
常规解法:

 struct ListNode* reverseList(struct ListNode* head) {
    struct ListNode new_head,*p = head,*q;
	new_head.next = NULL;
	while(p){
			q = p->next;
			p->next = new_head.next;
            new_head.next = p;
			p=q;
	}
	return new_head.next;
    }

递归解法:

 struct ListNode* reverseList(struct ListNode* head) {
     if(head == NULL||head->next == NULL)
            return head;
        struct ListNode *tail = head->next;
        struct ListNode *new_head = reverseList(head->next);
        head->next = tail->next;
        tail->next = head ;
        return new_head;
    }

3、双向链表

双向链表是在单链表的每个结点中,再设置一个指向其前驱结点的指针域。所以在双向链表中的结点都有两个指针域,一个指向直接后继,另一个指向直接前驱。

typedef struct DoubleLinkNode
{
		int data;//数据域
		struct DoubleLinkNode *pre; //指向前一个的地址
		struct DoubleLinkNode *next;//指向后一个的地址
}DulNode;

在这里插入图片描述
双向链表的优点是可以找到前驱和后继,可进可退。
但是,这同时也让增加和删除变得复杂,需要多分配一个指针存储空间。

4、循环链表

循环链表是指在链表的基础上,表的最后一个元素指向链表头结点,不再是为空。
那么头指针指向那个节点呢?
在循环链表中,头指针指向的是循环链表的最后一个节点。
因为在循环链表中,最后一个节点即类似于前面提到的虚拟头节点,也是一个真实的节点。

Ⅰ、单向循环链表

从一个结点出发可以找到其他任何一个结点。
在这里插入图片描述

Ⅱ、双向循环链表

表头结点的 pre指向表尾结点;表尾结点的next指向头结点。
双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。
在这里插入图片描述

Ⅲ、循环链表总结

在这里插入图片描述

Ⅳ、一些OJ题

①、环形链表

力扣-141:环形链表
给你一个链表的头节点 head ,判断链表中是否有环。
如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。注意:pos 不作为参数进行传递 。仅仅是为了标识链表的实际情况。
如果链表中存在环,则返回 true 否则,返回false
在这里插入图片描述
使用快慢指针法,分别定义两个指针,从头结点出发,快的指针每次移动两个节点,慢的指针每次移动一个节点,如果 快慢指针指针在途中相遇 ,说明这个链表有环。

bool hasCycle(struct ListNode *head) {
    struct ListNode *p = head,*q=head;
        while(q&&q->next)
        {
            p = p->next;
            q = q->next->next;
            if(p==q)
                return true;
        }
        return false;
}

②、快乐数

力扣-202:快乐数
编写一个算法来判断一个数 n 是不是快乐数。
「快乐数」 定义为:
对于一个正整数,每一次将该数替换为它每个位置上的数字的平方和。
然后重复这个过程直到这个数变为 1,也可能是 无限循环 但始终变不到 1。
如果这个过程 结果为 1,那么这个数就是快乐数。
如果 n 是 快乐数 就返回 true;不是,则返回 false

在这里插入图片描述
可以转化为上面的找环问题。

 int getNext(int x) {
        int d, y = 0;
        while (x) {
            d = x % 10;
            y += d * d;
            x /= 10;
        }
        return y;
    }
    bool isHappy(int n) {
        int p = n, q = n;
        while (q != 1) {
            p = getNext(p);
            q = getNext(getNext(q));
            if (p == q && p != 1) return false;
        }
        return true;
    }

三、总结

1、区别

不同点顺序表链表
存储空间上物理上一定连续逻辑上连续,但物理上不一定连续
随机访问支持:O(1)不支持:O(N)
任意位置插入或者删除元素可能需要搬移元素,效率低O(N)只需修改指针指向
插入动态顺序表,空间不够时需要扩容没有容量的概念
应用场景元素高效存储+频繁访问任意位置插入和删除频繁
缓存利用率

2、优点

①.链表是一个动态数据结构,无需给出链表的初始大小。
②.任意位置插入删除时间复杂度为O(1)。
与数组不同,在插入或删除元素后,我们不必移动元素。 在链表中,我们只需要更新节点的下一个指针中存在的地址即可。
③.由于链表的大小可以在运行时增加或减小,因此不会浪费内存。

3、缺点

①.存储密度小,因为每个数据元素,都需要额外存储一个指向下一元素的指针(双链表则需要两个指针)。
②.要访问特定元素,只能从链表头开始,遍历到该元素,时间复杂度为 O ( n) 在特定的数据元素之后插入或删除元素,不涉及到其他元素的移动,因此时间复杂度为 O ( 1) 。 双链表还允许在特定的数据元素之前插入或删除元素。
③.存储空间不连续,数据元素之间使用指针相连,每个数据元素只能访问周围的一个元素(根据单链表还是双链表有所不同)。

在这里插入图片描述

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

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

相关文章

云his系统源码 SaaS应用 基于Angular+Nginx+Java+Spring开发

云his系统源码 SaaS应用 功能易扩 统一对外接口管理 一、系统概述&#xff1a; 本套云HIS系统采用主流成熟技术开发&#xff0c;软件结构简洁、代码规范易阅读&#xff0c;SaaS应用&#xff0c;全浏览器访问前后端分离&#xff0c;多服务协同&#xff0c;服务可拆分&#xff…

【Linux要笑着学】进程创建 | 进程终止 | slab分派器

爆笑教程《看表情包学Linux》&#x1f448; 猛戳订阅&#xff01;​​​​​​​​​​​​&#x1f4ad; 写在前面&#xff1a;本章我们主要讲解进程的创建与终止。首先讲解进程创建&#xff0c;fork 函数是我们早在讲解 "进程的概念" 章节就提到过的一个函数&#…

总结篇 字符串设备(一)

简介 1、字符设备是Linux驱动中最基本的一类设备驱动&#xff0c;字符设备就是一个个字节&#xff0c;按照字节流进行读写操作的设备。&#xff08;例&#xff1a;按键&#xff0c;电池等&#xff0c;IIC,SPI&#xff0c;LCD&#xff09;。这些设备的驱动就叫字符设备驱动。 在…

八股文(二)

一、 实现深拷贝和浅拷贝 1.深拷贝 function checkType(any) {return Object.prototype.toString.call(any).slice(8, -1) }//判断拷贝的要进行深拷贝的是数组还是对象&#xff0c;是数组的话进行数组拷贝&#xff0c;对象的话进行对象拷贝 //如果获得的数据是可遍历的&#…

小白推荐!必定成功的python的安装流程?

目录 1.安装教程 2.使用cmd测试是否安装成功&#xff0c;快捷键WinR 3.如果测试失败&#xff0c;如何卸载&#xff1f; 4.如何在pycharm中指定下载的python解释器路径&#xff1f; 5.第一条python语句 1.安装教程 1.前往python的官网&#xff08;弄个梯子可能会快一点&#xf…

spring boot + rabbitMq整合之死信队列(DL)

rabbit mq 死信队列 什么是死信队列? DL-Dead Letter 死信队列 死信&#xff0c;在官网中对应的单词为“Dead Letter”&#xff0c;可以看出翻译确实非常的简单粗暴。那么死信是个什么东西呢&#xff1f; “死信”是RabbitMQ中的一种消息机制&#xff0c;当你在消费消息时&…

Qt 解决程序全屏运行弹窗引发任务栏显示

文章目录摘要在VM虚拟机器中测试setWindowFlags()关键字&#xff1a; Qt、 Qt::WindowStayOnTopHint、 setWindowFlags、 Qt::Window、 Qt::Tool摘要 今天眼看项目就要交付了&#xff0c;结果在测试程序的时候&#xff0c;发现在程序全品情况下&#xff0c;点击输入框&#x…

【Android Studio】【学习笔记】【2023春】

文章目录零、常用一、界面布局疑问&报错零、常用 一、界面布局 Android——六大基本布局总结/CSDN小马 同学 【Android】线性布局&#xff08;LinearLayout&#xff09;最全解析/CSDNTeacher.Hu 一个不错的计算器界面&#x1f447; Android Studio App LinearLayout多层…

数据资产管理建设思考(二)

关于数据资产管理&#xff0c;近两年是数据治理行业中一个热点话题&#xff0c;当然有我们前面提到的国家的政策支持及方向指引的原因。另一方面我们做数据治理的同行们从学习吸收国外优秀的数据治理理论&#xff0c;进一步在实践中思考如何应用理论&#xff0c;并结合我们国家…

docker(二)镜像详解、镜像构建、镜像优化

文章目录前言一、docker镜像详解1.镜像分层结构2.镜像的表示二、镜像构建1.commit提交2.DockerfileDockerfile 命令详解三、镜像优化1.缩减镜像层2.多阶段构建3.使用最精简的基础镜像前言 一、docker镜像详解 1.镜像分层结构 共享宿主机的kernelbase镜像提供的是最小的Linux发…

【LeetCode】1487. 保证文件名唯一

1487. 保证文件名唯一 题目描述 给你一个长度为 n 的字符串数组 names 。你将会在文件系统中创建 n 个文件夹&#xff1a;在第 i 分钟&#xff0c;新建名为 names[i] 的文件夹。 由于两个文件 不能 共享相同的文件名&#xff0c;因此如果新建文件夹使用的文件名已经被占用&a…

pytorch-模型训练中过拟合和欠拟合问题。从模型复杂度和数据集大小排查问题

评价了机器学习模型在训练数据集和测试数据集上的表现。如果你改变过实验中的模型结构或者超参数&#xff0c;你也许发现了&#xff1a;当模型在训练数据集上更准确时&#xff0c;它在测试数据集上却不一定更准确。这是为什么呢&#xff1f; 训练误差和泛化误差 在解释上述现象…

常用Swagger注解汇总

常用Swagger注解汇总 前言 在实际编写后端代码的过程中&#xff0c;我们可能经常使用到 swagger 注解&#xff0c;但是会用不代表了解&#xff0c;你知道每个注解都有什么属性吗&#xff1f;你都用过这些属性吗&#xff1f;了解它们的作用吗&#xff1f;本文在此带大家总结一下…

6-2 SpringCloud快速开发入门:声明式服务消费 Feign实现消费者

声明式服务消费 Feign实现消费者 使用 Feign实现消费者&#xff0c;我们通过下面步骤进行&#xff1a; 第一步&#xff1a;创建普通 Spring Boot工程 第二步&#xff1a;添加依赖 <dependencies><!--SpringCloud 集成 eureka 客户端的起步依赖--><dependency>…

图解LeetCode——剑指 Offer 34. 二叉树中和为某一值的路径

一、题目 给你二叉树的根节点 root 和一个整数目标和 targetSum &#xff0c;找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径。叶子节点 是指没有子节点的节点。 二、示例 2.1> 示例 1&#xff1a; 【输入】root [5,4,8,11,null,13,4,7,2,null,null,5,1], t…

从 ChatGPT 爆火回溯 NLP 技术

ChatGPT 火遍了全网&#xff0c;多个话题频频登上热搜。见证了自然语言处理&#xff08;NLP&#xff09;技术的重大突破&#xff0c;体验到通用技术的无限魅力。GPT 模型是一种 NLP 模型&#xff0c;使用多层变换器&#xff08;Transformer&#xff09;来预测下一个单词的概率分…

cuda编程以及GPU基本知识

目录CPU与GPU的基本知识CPU特点GPU特点GPU vs. CPU什么样的问题适合GPU&#xff1f;GPU编程CUDA编程并行计算的整体流程CUDA编程术语&#xff1a;硬件CUDA编程术语&#xff1a;内存模型CUDA编程术语&#xff1a;软件线程块&#xff08;Thread Block&#xff09;网格&#xff08…

新界面Moonbeam DApp上线,替你先尝试了一番!

作者&#xff1a;充电中的小恐龙 请注意&#xff0c;本篇内容来自Moonbeam社区成员的无偿分享&#xff0c;与Moonbeam官方和Moonbeam中文社区无关。本文内容仅供参考&#xff0c;对于内容的准确性和实效性&#xff0c;请自行谨慎判断。 本文撰写于DApp上线Beta版本之时&#…

数据库开发(一文概括mysql基本知识)

Mysql 是最流行的关系型数据库管理系统&#xff0c;在 WEB 应用方面 MySQL 是最好的 关系型数据库(Relational Database Management System&#xff1a;关系数据库管理系统)应用软件之一。mysql在问开发中&#xff0c;几乎必不可少&#xff0c;因为其他的可能是要收费的&#x…

【运维】Linux定时任务 定时执行脚本

【运维】Linux定时任务 定时执行脚本 在安装完成操作系统后&#xff0c;默认会安装 crond 服务工具&#xff0c;且 crond 服务默认就是自启动的。crond 进程每分钟会定期检查是否有要执行的任务&#xff0c;如果有&#xff0c;则会自动执行该任务。 五分钟执行一次sh脚本 进入编…