数据结构-----【链表:基础】

链表基础

1、链表的理论基础

1)基础:

链表:通过指针串联在一起的线性结构,每个节点由两部分组成,一个是数据域,一个是指针域(存放指向下一个节点的指针),最后一个指针域指向null(空指针的意思)。

链表的入口节点称为链表的头结点也就是head。

在这里插入图片描述

2)链表类型:

链表包括单链表、双链表、循环链表

单链表:指针域只能指向节点的下一个节点。

双链表:既可以向前查询也可以向后查询。

在这里插入图片描述

循环链表:链表首尾相连,循环链表可以解决约瑟夫环的问题。

在这里插入图片描述

3)链表的存储方式:

​ 数组在内存中是连续分布的,但是链表在内存中并不是连续分布的,链表通过指针域的指针连接在内存中的各个节点。例如:这个链表起始节点为2, 终止节点为7, 各个节点分布在内存的不同地址空间上,通过指针串联在一起。

在这里插入图片描述

4)链表的定义:

这个单向链表中,正是因为我们定义了节点的构造函数,指明了可以把x丢给val,才可以在初始化的时候直接赋值。

//单链表
struct ListNode{
    int val;//数据
    ListNode* next;//指向下一个节点的指针
    ListNode(int x):val(x),next(NULL){} //节点构造函数
};
//为什么要自己写构造函数呢?c++可以自己生成构造函数
//通过自己定义构造函数初始化节点
ListNode* head = new ListNode(5);
//使用默认构造函数初始化节点,不能直接给变量赋值!!
ListNode* head = new ListNode;
head->val = 5;

5)链表的操作:

链表中要注意的就是是否更改原来指针!由项目引发的思考:不得不引入值传递和指针传递:

**传递值:**如果传递的是基本数据类型或结构体(而不是指针),则函数内对形参的修改不会影响外部的实际参数。

#include <stdio.h>

void modifyValue(int x) {
    x = x + 1;  // 修改的是 x 的副本,不会影响外部的实际参数
}

int main() {
    int a = 5;
    modifyValue(a);
    printf("%d\n", a);  // 输出 5,a 没有被修改
    return 0;
}

**传递指针:**如果传递的是指针,函数内对指针所指向的数据的修改会影响到外部的实际参数。但修改指针本身的值不会影响外部的指针。

#include <stdio.h>

void modifyPointer(int* ptr) {
    *ptr = *ptr + 1;  // 修改指针所指向的数据,会影响外部的实际参数
    ptr ++;      // 修改指针本身的值,不会影响外部的实际参数
}

int main() {
    int b = 10;
    int* p = &b;
    modifyPointer(p);
    printf("%d\n", *p);  // 输出 11,p 所指向的数据被修改
    return 0;
}

  • 递值时,函数内的修改不会影响外部的实际参数。

  • 传递指针时,函数内对指针所指向的数据的修改会影响外部的实际参数,但修改指针本身的值不会影响外部的指针。

1.打印链表

为了保险起见,还是可以在函数里面用临时变量保存链表值:

void SlistPrint(ListNode *phead){
    ListNode *cur = phead;
    while(!cur){
        printf("%d->", cur->data);
        cur = cur-> next;
    }
    printf("NULL\n");
}
2.清空链表:

好家伙不传入**pehead就要报错:

在这里插入图片描述

//一个结点的定义
typedef struct ListNode{
        int val;
        ListNode* next;
        //ListNode(int x):val(x),next(NULL){};//重构函数
}ListNode;


//因为要改变外部链表头的值,所以要传入**
//pphead 表示一个 ListNode 结构体的对象,而不是一个指针。在你的 SListClear 函数中,*pphead = NULL; 尝试将一个结构体对象设置为 NULL,这是不合法的。
//如果你的目的是通过函数清空链表并将外部传递的链表头指针设置为 NULL,你应该使用指向指针的指针,即 ListNode** pphead,并在函数内部通过解引用两次来修改外部传递的链表头指针的值。
void SListClear(ListNode **pphead)//**PPhead指向指针的指针
{
      ListNode *cur = *pphead;
      while(!cur){
          ListNode* temp = cur->next;
          free(cur);
          cur = temp;
      }
     *pphead = NULL;
}

int main(){
	ListNode* head = (ListNode*)malloc(sizeof(ListNode));
    head->val = 1;
    head->next = (ListNode*)malloc(sizeof(ListNode));
    head->next->val = 2;
    head->next->next = NULL;
	SListClear(&head);//注意这里是&相当于 **head
	  // 在这里 head 已经被设置为 NULL,链表被清空
    if (head == NULL) {
        printf("List is empty\n");
    }

    system("pause"); 
    return 0;
}

3.创建结点:
typedef struct ListNode{
        int val;
        ListNode* next;
        ListNode(int x):val(x),next(NULL){};//重构函数 C++可以直接写重构函数
}ListNode;

//等效于直接 ListNode* node = new ListNode(val);
ListNode* CreateListNode(int x){
    ListNode* NewNode = (ListNode*)malloc(sizeof(ListNode));
    ListNode->val = x;
    ListNode->next = NULL;
    return NewNode;
}


3.删除节点:

比如删除D节点:

首先将C节点的next指针指向E,然后再手动释放D节点(py有自己的内存回收机制,不用手动释放,但是c/c++最好手动释放)

特别注意:因为单链表的只能指向下一个节点,删除某个节点的时候指针是在这个节点前一个节点的位置的。

在这里插入图片描述

typedef struct ListNode{
        int val;
        ListNode* next;
        ListNode(int x):val(x),next(NULL){};
}ListNode;

ListNode* delete(ListNode * node){
	if(node->next!=NULL && node->next->next!=NULL){
		node->next = node->next->next;
	}
	return node;
}
4、添加节点:

​ 在C和D中添加节点,让C的指针域指向F,再把F的指针域指向D。(添加不需要释放内存)

​ 链表的增添和删除都是O(1)操作,也不会影响到其他的节点,但是查找的时间复杂度可能是O(n)(一个一个next指针进行查找)。

typedef struct ListNode{
        int val;
        ListNode* next;
        ListNode(int x):val(x),next(NULL){};
}ListNode;

ListNode* add(ListNode * node,int val){
	ListNode* tmp = new ListNode(val);
	temp->next = node->next;
	node->next = temp;
	return node;
}

4.数组和链表的对比

在这里插入图片描述

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

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

相关文章

2024年必备的15个免费 SVG 设计资源

在动态设计领域&#xff0c;SVG&#xff08;可缩放矢量图形&#xff09;已成为设计师打造响应迅速、清晰且适应性强的视觉效果的必备工具。 这些设计非常适合幻灯片 PowerPoint 演示文稿、应用程序设计、网站设计、原型设计、社交媒体帖子等。 在这篇文章中&#xff0c;我们将…

LeetCode刷题之HOT100之最小栈

听歌&#xff0c;做题&#xff01; 1、题目描述 2、逻辑分析 拿到题目一脸懵&#xff0c;有点看不懂啥意思&#xff0c;看了题解才知道啥意思。要实现在常数时间内检索到最小元素的栈&#xff0c;需要使用一个辅助栈来每次存入最小值。 使用Deque作为栈的实现是因为它提供了…

最热门的智能猫砂盆好不好用?这期统统告诉你!

身为上班族的我们&#xff0c;常常被工作和出差填满日程。忘记给猫咪铲屎也不是一次两次了。但我们必须意识到&#xff0c;不及时清理猫砂盆不仅会让猫咪感到不适&#xff0c;还可能引发泌尿系统感染、皮肤疾病等健康问题。为了解决这个问题&#xff0c;越来越多的铲屎官开始将…

Unity数据持久化3——Json

概述 基础知识 Json文件格式 Json基本语法 练习 可以搜索&#xff1a;Json在线&#xff0c;复制代码进去解析是否写错了。 Excel转Json C#读取存储Json文件 JsonUtility using System.Collections; using System.Collections.Generic; using System.IO; using UnityEngine;[Sy…

使用Birdeye访问Sui上加密市场数据

是一个链上加密交易数据聚合器&#xff0c;于2024年4月开始整合Sui数据。 个人DeFi用户可以在Birdeye的首页找到丰富的数据&#xff0c;包括关于主流区块链上的tokens、交易和交易者钱包的详细信息。 Birdeye提供API和WebSockets数据服务&#xff0c;涵盖token价格和其他DeFi…

【jupyter notebook】解决打不开以及安装扩展插件的问题

文章目录 问题描述问题 1解决问题 2解决 问题描述 问题 1 在自定义的虚拟环境下&#xff0c;安装 jupyter notebook 6.4.12 版本时&#xff0c;报以下错误&#xff1a; 解决 查了一些 解决方法&#xff0c;执行以下命令即可解决&#xff1a; conda install traitlets5.9.0 …

PS系统教程27

Photoshop与Camera Raw Camera本身是作为插件存在的&#xff0c;处理对象Raw格式&#xff08;高清格式的图片标准&#xff09; JPG是压缩格式 Camera是源数据包&#xff0c;无损高清数据包 通道 通道只有黑白灰三种颜色&#xff0c;图层类似于前台&#xff0c;通道就是后台…

代码随想录算法训练营第七十天 | 108.冗余连接、109.冗余连接||

108.冗余连接 文字讲解&#xff1a;108. 冗余连接 | 代码随想录 解题思路 节点A 和节点 B 不在同一个集合&#xff0c;那么就可以将两个 节点连在一起 已经判断 节点A 和 节点B 在在同一个集合&#xff08;同一个根&#xff09;&#xff0c;如果将 节点A 和 节点B 连在一起就…

LabVIEW与PLC通讯方式及比较

LabVIEW与PLC之间的通讯方式多样&#xff0c;包括使用MODBUS协议、OPC&#xff08;OLE for Process Control&#xff09;、Ethernet/IP以及串口通讯等。这些通讯方式各有特点&#xff0c;选择合适的通讯方式可以提高系统的效率和稳定性。以下将详细介绍每种通讯方式的特点、优点…

Typora自动保存和找回未保存文件

在用typora做记录的时候没有手动保存&#xff0c;然后电脑崩了&#xff0c;还好有找回未保存文件功能&#xff0c;在这里存一下。 找到未保存的文件版本后将其内容复制到新文件即可。

【AI落地应用实战】如何高效检索与阅读论文——302.AI学术论文工具评测

一、引言 作为一名学术领域的探索者&#xff0c;我们都知道&#xff0c;检索和阅读论文是我们获取知识、启发思考、验证假设的基石&#xff0c;也是日常学习中必不可少的基本功之一。然而在浩瀚的学术海洋中&#xff0c;如何快速、准确地找到我们需要的论文&#xff0c;就像是…

Rust编写测试及控制执行

编写测试及控制执行 在 Rust 中&#xff0c;测试是通过函数的方式实现的&#xff0c;它可以用于验证被测试代码的正确性。测试函数往往依次执行以下三种行为&#xff1a; 设置所需的数据或状态运行想要测试的代码判断( assert )返回的结果是否符合预期 让我们来看看该如何使…

这几个PR小技巧你Get到了吗?

学习是永无止境的&#xff0c;需要不间断地学习&#xff0c;获取新知识。今天带来了5个PR小技巧&#xff0c;可以先收藏起来Adobe Premiere Pro 2024的获取查看Baidu Cloud 1、双倍稳定画面更舒适 一般来说大型电视剧、电影使用的拍摄设备都是非常高端的&#xff0c;不像我们…

Chrome插件:​Vue.js Devtools 高效地开发和调试

在现代前端开发中&#xff0c;Vue.js因其灵活性和性能优势&#xff0c;受到越来越多开发者的青睐。然而&#xff0c;随着项目规模的扩大&#xff0c;调试和优化变得愈发复杂。幸运的是&#xff0c;Vue.js Devtools的出现&#xff0c;为开发者提供了一套强大的工具集&#xff0c…

Unity 弧形图片位置和背景裁剪

目录 关键说明 Unity 设置如下 代码如下 生成和部分数值生成 角度转向量 计算背景范围 关键说明 效果图如下 来自红警ol游戏内的截图 思路&#xff1a;确定中心点为圆的中心点 然后 计算每个的弧度和距离 Unity 设置如下 没什么可以说的主要是背景图设置 代码如下 …

【Deep Learning】Self-Supervised Learning:自监督学习

自监督学习 本文基于清华大学《深度学习》第12节《Beyond Supervised Learning》的内容撰写&#xff0c;既是课堂笔记&#xff0c;亦是作者的一些理解。 在深度学习领域&#xff0c;传统的监督学习(Supervised Learning)的形式是给你输入 x x x和标签 y y y&#xff0c;你需要训…

Vue3 国际化i18n

国际化i18n方案 1. 什么是i18n2. i18n安装、配置及使用2.1 安装2.2 配置2.3 挂载到实例2.4 组件中使用2.5 语言切换 1. 什么是i18n i18n 是“国际化”的简称。在资讯领域&#xff0c;国际化(i18n)指让产品&#xff08;出版物&#xff0c;软件&#xff0c;硬件等&#xff09;无…

数据库系统体系结构-DBMS的三级模式结构、DBMS的工作方式、模式定义语言、二级映射

一、体系结构的概念 1、大多数DBMS遵循三级模式结构 &#xff08;1&#xff09;外模式 &#xff08;2&#xff09;概念模式 &#xff08;3&#xff09;内模式 2、DBMS的体系结构描述的应该是系统的组成结构及其联系以及系统结构的设计和变化的原则等 3、1978年美国国家标…

双向长短期记忆神经网络BiLSTM

先说一下LSTM LSTM 是一种特殊的 RNN&#xff0c;它通过引入门控机制来解决传统 RNN 的长期依赖问题。 LSTM 的结构包含以下几个关键组件&#xff1a; 输入门&#xff08;input gate&#xff09;&#xff1a;决定当前时间步的输入信息对细胞状态的影响程度。遗忘门&#xff…

大模型回归实业,少谈梦,多赚钱

前言 大家都知道美国现在AI很火&#xff0c;但是现在火到已经有点看不懂的地步了。 苹果前脚在WWDC24上公布了自己在AI上的新进展&#xff0c;隔天市值就上涨了2142亿美元。而以微软为首的美股“Big 7”的市值更是达到史无前例的14万亿&#xff0c;占据标普500的32%。 冷静下…