当前位置: 首页 > news >正文

数据结构与算法-双向链表专题

目录

一. 双向链表的结构

二.双向链表的使用

2.1 创建节点

2.2 初始化

2.3 打印

2.4 尾插

2.5 头插

2.6 尾删

2.7 头删

2.8 在指定位置pos之后插入数据

2.9 查找数据

2.10 删除pos位置的节点

2.11 销毁链表


一. 双向链表的结构

在List.h的头文件中对链表的结构进行创建

#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
typedef int LTDataType;//双向链表的结构
typedef struct ListNode
{LTDataType data;struct ListNode* next;struct ListNode* prev;
}LTNode;

为了方便使用,将结构体类型重命名为LTNode

将存储数据的类型通过typedef来使用,方便进行修改

如果双向链表为空,则只有头节点一个节点。

如果phead=NULL,则只能说明这不是一个有效的双向链表。

二.双向链表的使用

2.1 创建节点

(下列的声明就不再显示,直接写函数的实现)

//创建节点
LTNode* LTNBuyNode(LTDataType x)
{LTNode* node = (LTNode*)malloc(sizeof(LTNode));if (node == NULL){perror("malloc fail!");exit(1);}node->data = x;node->next = node->prev = node;return node;
}

 由于带头双向循环链表的特性,prev和next都指向node本身,也就是自身循环

2.2 初始化

//初始化
void LTInit(LTNode** phead)
{* phead =LTNBuyNode(-1);
}

初始化就是创建哨兵位哨兵位的数据和地址都不会被修改

哨兵位也就是头节点,原先单链表中说的头节点其实是指第一个有效节点。

第二种方式:

LTNode* LTInit()
{LTNode*phead = LTNBuyNode(-1);return phead;
}

 无需创建变量,而是直接返回phead

2.3 打印

//打印
void LTPrint(LTNode* phead)//int 类型
{LTNode* pcur = phead->next;while (pcur != phead){printf("%d->", pcur->data);pcur = pcur->next;}printf("\n");
}

2.4 尾插

//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = LTNBuyNode(x);newnode->next = phead;newnode->prev = phead->prev;phead->prev->next = newnode;phead->prev = newnode;
}

尾插主要是要直到双向链表尾插的结构和指针指向就很简单了

如果你要在哨兵位前面头插,那相当于尾插(因为双向链表是带头双向循环链表)

2.5 头插

//头插
void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = LTNBuyNode(x);newnode->next = phead->next;newnode->prev = phead;phead->next->prev = newnode;phead->next = newnode;
}

2.6 尾删

//尾删
void LTPopBack(LTNode* phead)
{//链表必须有效,且链表不为空assert(phead&&phead->next!=phead);LTNode* del = phead->prev;del->prev->next = phead;phead->prev = del->prev;free(del);del = NULL;
}

注意,需要del来承载phead->next

链表有效且不为空

2.7 头删

//头删
void LTPopFront(LTNode* phead)
{//链表必须有效,且链表不为空assert(phead && phead->next != phead);LTNode* del = phead->next;//phead,del,del->nextphead->next = del->next;del->next->prev = phead;free(del);del = NULL;
}

2.8 在指定位置pos之后插入数据

//在pos位置之后插⼊数据 
void LTInsert(LTNode* pos, LTDataType x)
{assert(pos);LTNode* newnode = LTNBuyNode(x);newnode->next = pos->next;newnode->prev = pos;pos->next->prev = newnode;pos->next = newnode;
}

2.9 查找数据


//查找
LTNode* LTFind(LTNode* phead, LTDataType x)
{LTNode* pcur = phead->next;while (pcur != phead){if (pcur->data == x){return pcur;}pcur = pcur->next;}return NULL;
}

注意返回的是数据的节点地址

所以之后要检验find是否为空来判断是否有

2.10 删除pos位置的节点

//删除pos节点
//为什么不传二级指针
//为了保证接口的一致性void LTErase(LTNode* pos)
{//理论上pos不能为phead,但是没有参数phead,无法增加校验assert(pos);//pos,pos->next,pos->prevpos->next->prev = pos->prev;pos->prev->next = pos->next;free(pos);pos = NULL;
}

这里不使用二级指针传参是因为为了保证接口的统一性,因为之前的形参都是一级指针

所以此函数使用后需要将find指针变为NULL

2.11 销毁链表

//销毁链表
void LTDestroy(LTNode* phead)
{assert(phead);LTNode* pcur = phead->next;while (pcur != phead){LTNode* next = pcur->next;free(pcur);pcur = next;}free(phead);phead = NULL;
}

使用后还得将plist(实参)赋值为空指针

原来还是应该传二级指针,但是为了接口统一性,跟前者一样

相关文章:

  • 如何自定义 Spring MVC 的配置?
  • [学习] RTKLib详解:rtcm2.c、rtcm3.c、rtcm3e与rtcmn.c
  • Spring Web MVC————入门(2)
  • 【极兔快递Java社招】一面复盘|数据库+线程池+AQS+中间件面面俱到
  • 下载的旧版的jenkins,为什么没有旧版的插件
  • 【docker】--容器管理
  • 代码随想录训练营第二十三天| 572.另一颗树的子树 104.二叉树的最大深度 559.N叉树的最大深度 111.二叉树的最小深度
  • 入门OpenTelemetry——部署OpenTelemetry
  • AI智能分析网关V4周界入侵检测算法精准监测与智能分析,筑牢周界安全防线
  • 《AI大模型应知应会100篇》第62篇:TypeChat——类型安全的大模型编程框架
  • 【SSL证书系列】操作系统如何保障根证书的有效性和安全
  • 适配华为昇腾 NPU 的交互式监控工具
  • 大模型训练简介
  • json-server的用法-基于 RESTful API 的本地 mock 服务
  • 仿射变换 与 透视变换
  • 第二个五年计划!
  • 计算机网络:手机和基站之间是通过什么传递信息的?怎么保证的防衰减,抗干扰和私密安全的?
  • Java内存马的检测与发现
  • 基于 GPUGEEK平台进行vLLM环境部署DeepSeek-R1-70B
  • 一分钟在Cherry Studio和VSCode集成火山引擎veimagex-mcp
  • 昆明警方重拳打击经济领域违法犯罪:去年抓获905名嫌犯
  • 欧元区财长会讨论国际形势及应对美国关税政策
  • 人民日报访巴西总统卢拉:“巴中关系正处于历史最好时期”
  • 中美经贸高层会谈在瑞士日内瓦举行
  • 海航回应“男团粉丝为追星堵住机舱通道”:已紧急阻止
  • 印度一战机在巴基斯坦旁遮普省被击落,飞行员被俘