本文介绍: 世上有两种耀眼的光芒,一种是正在升起的太阳,一种是正在努力学习编程的你!一个爱学编程的人。各位看官,我衷心的希望这篇博客能对你们有所帮助,同时也希望各位看官能对我的文章给与点评,希望我们能够携手共同促进进步,在编程的道路上越走越远!提示:以下是本篇文章正文内容,下面案例可供参考List.h//定义双向链表的节点的结构//保存下一个节点的地址//保存前一个节点的地址}LTNode;//链表的初始化//前提:我们要传入一个头节点//我们更倾向于第二种初始化的方法。
前言
世上有两种耀眼的光芒,一种是正在升起的太阳,一种是正在努力学习编程的你!一个爱学编程的人。各位看官,我衷心的希望这篇博客能对你们有所帮助,同时也希望各位看官能对我的文章给与点评,希望我们能够携手共同促进进步,在编程的道路上越走越远!
1. 双向链表的结构
注意:这里的“带头”跟前面我们说的“头节点”是两个概念,实际前面的在单链表阶段称呼不严 谨,但是为了同学们更好的理解就直接称为单链表的头节点。
2. 双向链表的实现
2.1 头文件 ——双向链表的创建及功能函数的定义
#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
//定义双向链表的节点的结构
typedef int STDataType;
typedef struct ListNode
{
STDataType data;
struct ListNode* next;//保存下一个节点的地址
struct ListNode* prev;//保存前一个节点的地址
}LTNode;
//链表的初始化
//void LTInit(LTNode** pphead);//前提:我们要传入一个头节点
//我们更倾向于第二种初始化的方法
//因为双向链表为空(只有一个哨兵位:哨兵位节点是不能被操作的,即不能被改变)
LTNode* LTInit();//不需要传入参数,调用该方法之后给我们返回一个头节点
//在双向链表中不会改变哨兵位,所以可以传一级指针
//尾插入操作
void LTPushBack(LTNode* phead, STDataType x);
//头插
void LTPushFront(LTNode* phead, STDataType x);
//链表的打印
void LTPrint(LTNode* phead);
//尾删
void LTPopBack(LTNode* phead);
//头删
void LTPopFront(LTNode* phead);
//在pos位置之后插入的数据
void LTInsert(LTNode* pos, STDataType x);
//删除pos位置的节点
void LTErase(LTNode* pos);
LTNode* LTFind(LTNode* phead, STDataType x);
//链表的销毁
void LTDestroy(LTNode* phead);
2.2 源文件 ——双向链表的功能函数的实现
List.c
#define _CRT_SECURE_NO_WARNINGS 1
#include "List.h"
//链表的初始化
//前提:我们要传入一个头节点
//void LTInit(LTNode** pphead)
//{
// *pphead = (LTNode*)malloc(sizeof(LTNode));
// if (*pphead == NULL)
// {
// perror("malloc");
// return;
// }
// //节点有三部分内容:数据 前驱指针 后继指针
// (*pphead)->data = -1;//哨兵位
// (*pphead)->next = (*pphead)->prev = *pphead;
//}
//链表初始化
//不需要传入参数,调用该方法之后给我们返回一个头节点
LTNode* LTInit()
{
LTNode* phead = (LTNode*)malloc(sizeof(LTNode));
if (phead == NULL)
{
perror("malloc");
return;
}
phead->data = -1;
phead->next = phead->prev = phead;
return phead;
}
//申请一个新的节点
LTNode* ListBuyNode(STDataType x)
{
LTNode* node = (LTNode*)malloc(sizeof(LTNode));
node->data = x;
node->next = node->prev = NULL;
return node;
}
//链表的打印
void LTPrint(LTNode* phead)
{
LTNode* cur = phead->next;
while (cur != phead)
{
printf("%d->", cur->data);
cur = cur->next;
}
printf("n");
}
//尾插入操作
void LTPushBack(LTNode* phead, STDataType x)
{
assert(phead);
LTNode* node = ListBuyNode(x);
//先处理新节点node的前驱和后继指针
node->prev = phead->prev;
node->next = phead;
//在处理phead->prev(之前的尾节点)和phead
phead->prev->next = node;
phead->prev = node;
}
//头插
void LTPushFront(LTNode* phead, STDataType x)
{
assert(phead);
LTNode* node = ListBuyNode(x);
//node的节点 next prev
node->prev = phead;
node->next = phead->next;
//处理phead phead->next
phead->next->prev = node;
phead->next = node;
}
//尾删
void LTPopBack(LTNode* phead)
{
assert(phead);
//链表不能为空链表:链表中只有一个哨兵位节点
assert(phead->next != phead);
LTNode* del = phead->prev;
//先处理del->prev节点
del->prev->next = phead;
//处理phead
phead->prev = del->prev;
free(del);
del = NULL;
}
//头删
void LTPopFront(LTNode* phead)
{
assert(phead&& phead->next != phead);
LTNode* del = phead->next;
//phead del->next
del->next->prev = phead;
phead->next = del->next;
free(del);
del = NULL;
}
//在pos位置之后插入的数据
void LTInsert(LTNode* pos, STDataType x)
{
assert(pos);
LTNode* node = ListBuyNode(x);
//node的prev 和 next
node->next = pos->next;
node->prev = pos;
//pos的next 和 pos->next的prev
pos->next = node;
node->next->prev = node;
}
//删除pos位置的节点
void LTErase(LTNode* pos)
{
assert(pos);
//pos->prev:next pos pos->next:prev
pos->next->prev = pos->prev;
pos->prev->next = pos->next;
free(pos);
pos = NULL;
}
//查找数据
LTNode* LTFind(LTNode* phead, STDataType x)
{
assert(phead);
LTNode* cur = phead->next;
while (cur != phead)
{
if (cur->data == x)
{
return cur;
}
cur = cur->next;
}
return NULL;
}
//链表的销毁
void LTDestroy(LTNode* phead)
{
assert(phead);
LTNode* cur = phead->next;
while (cur != phead)
{
LTNode* next = cur->next;
free(cur);
cur = next;
}
//除了循环之后,还有哨兵位没有被释放
free(phead);
phead = NULL;
//我们将phead指向的空间释放掉,plist实参的空间也被释放掉了,phead置为空
//但是此时plist实参为野指针,还需要我们手动置为空
}
2.3 源文件 ——双向链表功能的测试
#define _CRT_SECURE_NO_WARNINGS 1
#include "List.h"
void ListTest()
{
//第一种初始化方法:
/*LTNode* plist = NULL;
LTInit(&plist);*/
//第二种初始化方法;
LTNode* plist = LTInit();
//尾插
LTPushBack(plist, 1);
LTPushBack(plist, 2);
LTPushBack(plist, 3);
LTPushBack(plist, 4);
//打印
LTPrint(plist);
头插
//LTPushFront(plist, 5);
//LTPushFront(plist, 6);
//LTPushFront(plist, 7);
//LTPrint(plist);
//尾删
//LTPopBack(plist);
//头删
/*LTPopFront(plist);
LTPopFront(plist);*/
//测试指定位置之后插入
//LTNode* find = LTFind(plist, 1);
//LTInsert(find, 11);
/*LTErase(find);
LTPrint(plist);*/
//销毁链表
LTDestroy(plist);
//传一级指针的要手动将plist置为空
plist = NULL;
}
int main()
{
ListTest();
return 0;
}
4.双向链表的操作示意图
3.顺序表和双向链表的优缺点分析
总结
好了,本篇博客到这里就结束了,如果有更好的观点,请及时留言,我会认真观看并学习。
不积硅步,无以至千里;不积小流,无以成江海。
原文地址:https://blog.csdn.net/2301_79585944/article/details/134722446
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.7code.cn/show_38780.html
如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱:suwngjj01@126.com进行投诉反馈,一经查实,立即删除!
声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。