数据结构(一)链表

目录

链表

单向链表

单向链表结构与基本操作

插入节点

删除节点

搜索节点

遍历链表

反转链表

双向链表

双向链表结构与基本操作

节点定义和创建

插入节点

删除节点

搜索节点

遍历链表

转链表反


在开始讲线性表之前,先给各位读者重新回顾一下链表

链表

单向链表

单向链表是一种基本的数据结构,由一系列节点组成,每个节点包含两部分:数据域指针域数据域存储数据,而指针域存储指向列表中下一个节点的指针。单向链表的特点是数据元素的排列呈现单一方向,即每个节点仅指向下一个节点,直到最后一个节点的指针通常指向NULL,表示链表的结束。

单向链表结构与基本操作

在C语言中,单向链表的节点可以这样定义

typedef struct Node {int data;                 // 数据域,存储节点的数据struct Node* next;        // 指针域,指向下一个节点的指针
} Node;

 创建节点:初始化一个节点,通常需要提供数据,节点的next指针设置为NULL

Node* createNode(int data) {Node* newNode = (Node*)malloc(sizeof(Node));if (newNode != NULL) {newNode->data = data;newNode->next = NULL;}return newNode;
}
插入节点

插入节点的基本过程:首先找到正确位置p,然后申请新结点t并对t的结点信息赋值,最后讲t插在p之后

插入到头部

void insertAtHead(Node** head, int data) {Node* newNode = createNode(data);newNode->next = *head;*head = newNode;
}

插入到尾部
在链表尾部插入节点需要遍历整个链表,直到找到最后一个节点。

void insertAtTail(Node** head, int data) {Node* newNode = createNode(data);if (*head == NULL) {*head = newNode;} else {Node* temp = *head;while (temp->next != NULL) {temp = temp->next;}temp->next = newNode;}
}

插入到特定位置

在链表的特定位置插入节点需要遍历链表直到达到指定位置的前一个节点。

void insertAtPosition(Node** head, int data, int position) {Node* newNode = createNode(data);if (position == 1) {insertAtHead(head, data);} else {Node* temp = *head;for (int i = 1; temp != NULL && i < position - 1; i++) {temp = temp->next;}if (temp == NULL) {printf("Position exceeds the length of the list.\n");} else {newNode->next = temp->next;temp->next = newNode;}}
}
删除节点

注意:删除一个节点后必须释放该节点的空间

删除头部节点

void deleteFromHead(Node** head) {if (*head != NULL) {Node* temp = *head;*head = (*head)->next;free(temp);}
}

删除尾部节点

删除尾部节点需要遍历整个链表以找到倒数第二个节点。

void deleteFromTail(Node** head) {if (*head == NULL) return;if ((*head)->next == NULL) {free(*head);*head = NULL;} else {Node* temp = *head;while (temp->next->next != NULL) {temp = temp->next;}free(temp->next);temp->next = NULL;}
}

删除特定位置的节点

删除特定位置的节点需要遍历链表直到找到该位置的前一个节点。

void deleteAtPosition(Node** head, int position) {if (*head == NULL) return;if (position == 1) {deleteFromHead(head);} else {Node* temp = *head;for (int i = 1; temp != NULL && i < position - 1; i++) {temp = temp->next;}if (temp == NULL || temp->next == NULL) {printf("Position exceeds the length of the list.\n");} else {Node* next = temp->next->next;free(temp->next);temp->next = next;}}
}
搜索节点

搜索节点涉及遍历链表以查找具有特定值的节点。

Node* search(Node* head, int key) {Node* current = head;while (current != NULL) {if (current->data == key) {return current;}current = current->next;}return NULL;
}
遍历链表

遍历链表是访问链表中每个节点的基本操作。

void printList(Node* head) {Node* current = head;while (current != NULL) {printf("%d -> ", current->data);current = current->next;}printf("NULL\n");
}
反转链表

反转链表需要改变每个节点的指针方向。

Node* reverseList(Node* head) {Node* prev = NULL;Node* current = head;Node* next = NULL;while (current != NULL) {next = current->next;current->next = prev;prev = current;current = next;}return prev;
}

双向链表

双向链表(Doubly Linked List)是一种链式数据结构,其中每个节点包含数据部分以及两个指针,分别指向前一个节点和后一个节点。这种结构允许从两个方向遍历链表:向前和向后。双向链表在插入和删除操作中比单向链表更加灵活,因为可以直接访问前驱和后继节点。

双向链表结构与基本操作

节点定义和创建

创建一个双向链表节点,需要初始化数据和两个指针:

typedef struct Node {int data;                 // 数据域,存储节点的数据struct Node* prev;        // 指向前一个节点的指针struct Node* next;        // 指向下一个节点的指针
} Node;Node* createNode(int data) {Node* newNode = (Node*)malloc(sizeof(Node));if (newNode != NULL) {newNode->data = data;newNode->prev = NULL;newNode->next = NULL;}return newNode;
}
插入节点

插入到头部

在头部插入节点,需要更新头节点的prev指针和新节点的next指针。

void insertAtHead(Node** head, int data) {Node* newNode = createNode(data);newNode->next = *head;if (*head != NULL) {(*head)->prev = newNode;}*head = newNode;
}

插入到尾部

在尾部插入节点,需要遍历链表找到尾节点,然后更新尾节点的next指针和新节点的prev指针。

void insertAtTail(Node** head, int data) {Node* newNode = createNode(data);if (*head == NULL) {*head = newNode;} else {Node* temp = *head;while (temp->next != NULL) {temp = temp->next;}temp->next = newNode;newNode->prev = temp;}
}

插入到特定位置

在特定位置插入节点,需要更新前一个节点的next指针和新节点的prev指针,以及新节点的next指针和后一个节点的prev指针。

void insertAtPosition(Node** head, int data, int position) {Node* newNode = createNode(data);if (position == 1) {insertAtHead(head, data);} else {Node* temp = *head;for (int i = 1; temp != NULL && i < position - 1; i++) {temp = temp->next;}if (temp == NULL) {printf("Position exceeds the length of the list.\n");} else {newNode->next = temp->next;newNode->prev = temp;if (temp->next != NULL) {temp->next->prev = newNode;}temp->next = newNode;}}
}
删除节点

删除头部节点:删除头部节点,需要更新头节点的prev指针和新头节点的next指针。

void deleteFromHead(Node** head) {if (*head == NULL) return;Node* temp = *head;*head = (*head)->next;if (*head != NULL) {(*head)->prev = NULL;}free(temp);
}

删除尾部节点:删除尾部节点,需要遍历链表找到倒数第二个节点,然后更新其next指针

void deleteFromTail(Node** head) {if (*head == NULL) return;if ((*head)->next == NULL) {free(*head);*head = NULL;} else {Node* temp = *head;while (temp->next->next != NULL) {temp = temp->next;}free(temp->next);temp->next = NULL;}
}

删除特定位置的节点:删除特定位置的节点,需要更新前一个节点的next指针和后一个节点的prev指针

void deleteAtPosition(Node** head, int position) {if (*head == NULL) return;if (position == 1) {deleteFromHead(head);} else {Node* temp = *head;for (int i = 1; temp != NULL && i < position; i++) {temp = temp->next;}if (temp == NULL) {printf("Position exceeds the length of the list.\n");} else {if (temp->prev != NULL) {temp->prev->next = temp->next;}if (temp->next != NULL) {temp->next->prev = temp->prev;}free(temp);}}
}
搜索节点

搜索节点,需要遍历链表直到找到具有特定值的节点。

Node* search(Node* head, int key) {Node* current = head;while (current != NULL) {if (current->data == key) {return current;}current = current->next;}return NULL;
}
遍历链表

向前遍历

void printListForward(Node* head) {Node* current = head;while (current != NULL) {printf("%d -> ", current->data);current = current->next;}printf("NULL\n");
}

向后遍历

void printListBackward(Node* head) {Node* current = head;while (current != NULL) {printf("%d <- ", current->data);current = current->prev;}printf("NULL\n");
}
转链表反
void reverseList(Node** head) {Node* temp = NULL;Node* current = *head;while (current != NULL) {temp = current->prev;current->prev = current->next;current->next = temp;current = current->prev;}if (temp != NULL) {*head = temp->prev;}
}

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

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

相关文章

linux从0到1——shell编程7

声明&#xff01; 学习视频来自B站up主 **泷羽sec** 有兴趣的师傅可以关注一下&#xff0c;如涉及侵权马上删除文章&#xff0c;笔记只是方便各位师傅的学习和探讨&#xff0c;文章所提到的网站以及内容&#xff0c;只做学习交流&#xff0c;其他均与本人以及泷羽sec团队无关&a…

微软发布Win11 24H2系统11月可选更新KB5046740!

系统之家11月22日报道&#xff0c;微软针对Win11 24H2系统推出2024年11月最新可选更新补丁KB5046740&#xff0c;更新后系统版本后升至26100.2454&#xff0c;此次更新后修复当应用程序以PDF和XLSX格式导出图表对象时停止响应、无法使用API查找旋转信息等问题。以下小编将给大家…

五天SpringCloud计划——DAY2之使用Docker完成项目的部署

一、引言 刚刚学完了Docker的使用&#xff0c;现在知识在脑子里面还是热乎的&#xff0c;是时候把它总结一下了。 现在的我认为Docker时一个部署项目的工具(不知道是不是真的),相比于我以前使用宝塔面板部署项目&#xff0c;使用Docker更能让我看到代码之美&#xff0c;怎么一…

.net6 使用 FreeSpire.XLS 实现 excel 转 pdf - docker 部署

FreeSpire.XLS && Aspose.Cells包都可以实现。实现过程中发现如下问题&#xff1a; 本地测试通过&#xff0c; docker部署服务器后报错&#xff1a; The type initializer for Spire.Xls.Core.Spreadsheet.XlsPageSetupBase threw an exception. 由于缺少依赖&#xf…

sed使用扩展正则表达式时, -i 要写在 -r 或 -E 的后面

sed使用扩展正则表达式时, -i 要写在 -r 或 -E 的后面 前言 -r 等效 -E , 启用扩展正则表达式 -E是新叫法,更统一,能增强可移植性 , 但老系统,比如 CentOS-7 的 sed 只能用 -r ### Ubuntu24.04-E, -r, --regexp-extendeduse extended regular expressions in the script(fo…

学习ASP.NET Core的身份认证(基于Cookie的身份认证2)

采用基于Cookie的身份认证&#xff0c;在调用services.AddAuthentication注册服务时&#xff0c;可以通过CookieAuthenticationOptions对象按需设置Cookie属性&#xff0c;常用的包括以下属性&#xff08;更详细的介绍见参考文献2&#xff0c;微软的帮助文档中的介绍看的头大&a…

设计模式之 模板方法模式

模板方法模式是行为型设计模式的一种。它定义了一个算法的骨架&#xff0c;并将某些步骤的实现延迟到子类中。模板方法模式允许子类在不改变算法结构的情况下重新定义算法的某些特定步骤。 模板方法模式的核心在于&#xff1a; 封装算法的骨架&#xff1a;通过父类中的模板方…

Softing线上研讨会 | Ethernet-APL:推动数字时代的过程自动化

| &#xff08;免费&#xff09;线上研讨会时间&#xff1a;2024年11月19日 16:00~16:30 / 23:00~23:30 Ethernet-APL以10Mb/s的传输速率为过程工业中的现场设备带来了无缝以太网连接和本质安全电源&#xff0c;这不仅革新了新建工厂&#xff0c;也适用于改造现有工厂。 与现…

Idea修改Commit Changes模式、idea使用git缺少部分Commit Changes

文章目录 一、模式一1、页面效果如下2、如何打开为这种样式&#xff1f; 二、模式二1、页面效果如下2、如何打开为这种样式&#xff1f; 三、总结 前言&#xff1a;Idea中代码提交到git库时的commit Change有两种模式&#xff0c;每种模式的界面及功能都不太一样。 Commit Cha…

Getx:GetxController依赖管理02,Binding绑定全局控制器(懒加载Controller)

在使用GetX 状态管理器的时候&#xff0c;如果每个页面都手动实例化一个控制器就太麻烦了&#xff0c; Binding 的作用就是所有需要进行状态管理的控制器进行统一初始化 创建全局控制器Binding import package:get/get.dart; import ../controllers/counter.dart; // 同上一篇内…

东土科技孵化的“网联汽车高速通信技术”前沿产品亮相2024WICV大会

2024世界智能网联汽车大会&#xff08;WICV&#xff09;于近日在北京召开。本次大会发布了由中国汽车工程学会组织全球200余位专家&#xff0c;联合评审遴选出未来十年对于智能网联汽车发展具有重要影响的十大技术趋势&#xff0c;包括“面向高级别自动驾驶的超级人工智能”“网…

torch.__version__的torch版本和conda list的torch版本不一致

conda list是1.9.0的torch&#xff0c;但是torch.__version__是1.6的&#xff0c;很奇怪 后来发现这个1.6的torch是base的版本 但是使用“python -m site”查看当前环境的包搜索路径&#xff0c;也是对的&#xff0c;先从我自己的虚拟环境搜索 最后从我自己的虚拟环境里pip uni…

nginx 配置lua执行shell脚本

1.需要nginx安装lua_nginx_module模块,这一步安装时&#xff0c;遇到一个坑&#xff0c;nginx执行configure时&#xff0c;一直提示./configure: error: unsupported LuaJIT version&#xff1b; ngx_http_lua_module requires LuaJIT 2.x。 网上一堆方法都试了&#xff0c;都…

【linux】插入新硬盘如何配置:格式化、分区、自动挂载(Ubuntu)

文章目录 具体方法GPT分区表&#xff08;GUID Partition Table&#xff09;&#xff08;建议都用这种分区方法&#xff09;MBR分区表方法&#xff08;最大支持2TB分区&#xff09;&#xff08;Master Boot Record&#xff09; 附加&#xff1a;如何查看硬盘的型号另外&#xff…

使用itextpdf进行pdf模版填充中文文本时部分字不显示问题

在网上找了很多种办法 都解决不了; 最后发现是文本域字体设置出了问题; 在这不展示其他的代码 只展示重要代码; 1 引入扩展包 <dependency><groupId>com.itextpdf</groupId><artifactId>itext-asian</artifactId><version>5.2.0</v…

【Ubuntu】如何在Ubuntu系统中查看端口是否可用

文章目录 前言一、使用netstat命令二、使用ss命令三、使用lsof命令四、使用nc&#xff08;netcat&#xff09;命令总结 前言 本文介绍了如何在Ubuntu系统中查看端口是否可用的方法&#xff0c;并给出了具体的命令示例&#xff0c;帮助用户通过命令行工具检测端口的开放状态。 …

长短时记忆网络(SLTM):理解与实践

长短时记忆网络&#xff08;SLTM&#xff09;&#xff0c;即短期长期记忆网络&#xff0c;是一种特殊的循环神经网络&#xff08;RNN&#xff09;&#xff0c;它能够学习到数据中的长期依赖关系。在这篇文章中&#xff0c;我们将详细介绍SLTM的工作原理&#xff0c;并提供一个简…

HTML5实现剪刀石头布小游戏(附源码)

文章目录 1.设计来源1.1 主界面1.2 皮肤风格1.2 游戏中界面 2.效果和源码源码下载万套模板&#xff0c;程序开发&#xff0c;在线开发&#xff0c;在线沟通 作者&#xff1a;xcLeigh 文章地址&#xff1a;https://blog.csdn.net/weixin_43151418/article/details/143798520 HTM…

自动驾驶之激光雷达

这里写目录标题 1 什么是激光雷达2 激光雷达的关键参数3 激光雷达种类4 自动驾驶感知传感器5 激光雷达感知框架5.1 pointcloud_preprocess5.2 pointcloud_map_based_roi5.3 pointcloud_ground_detection5.4 lidar_detection5.5 lidar_detection_filter5.6 lidar_tracking 1 什么…

pycharm在使用conda虚拟环境时Terminal爆红问题

问题&#xff1a; 解决方法&#xff1a; 复制cmd.exe后面所有路径 添加到pycharm的shell path中&#xff1a;