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

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

目录

一. 双向链表的结构

二.双向链表的使用

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(实参)赋值为空指针

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

http://www.xdnf.cn/news/6068.html

相关文章:

  • Spring AI 集成 Mistral AI:构建高效多语言对话助手的实战指南
  • 毕业论文,如何区分研究内容和研究方法?
  • C#中的dynamic与var:看似相似却迥然不同
  • Ota++框架学习
  • 胶片转场视频剪辑思路
  • tocmat 启动怎么设置 jvm和gc
  • 大模型训练简介
  • 华硕服务器-品类介绍
  • RBTree的模拟实现
  • MySQL之基础事务
  • 常用的应用层网络协议对比
  • 从零玩转系列之 MCP AI Agent 理论+项目实战开发你的MCP Server
  • UOS专业版上通过源码安装 Python 3.13 并保留系统默认版本
  • 310. 最小高度树
  • 『 测试 』软件测试全流程与Bug管理核心要点解析
  • 2025年6月一区SCI-不实野燕麦优化算法Animated Oat Optimization-附Matlab免费代码
  • OpenEvidence AI临床决策支持工具平台研究报告
  • 零成本打造专属AI图像处理平台:IOPaint本地部署与远程访问指南
  • 数据库系统概论|第六章:关系数据理论—课程笔记2
  • 嵌入式学习笔记 - SystemCoreClock/1000000
  • 自然语言处理入门级项目——文本分类
  • 多模态大语言模型arxiv论文略读(七十五)
  • 苍穹外卖 - Day02 学习笔记
  • vscode extention踩坑记
  • IP SSL怎么签发使用
  • DeepSearcher:开启智能搜索新纪元,赋能企业级数据研究
  • Prometheus+Grafana+AlertManager完整安装过程
  • UUG杭州站 | 团结引擎1.5.0 OpenHarmony新Feature介绍
  • 网络协议分析 实验七 FTP、HTTP、DHCP
  • ssti模板注入学习