挑战最详细注释 c语言 双链表 的实现

简介: 全代码实现,欢迎阅读!!

1.   定义创建并初始化

#include<stdio.h>          //引用头文件

#include<stdlib.h>

#include<assert.h>        

typedef struct Node      //定义一个结构体

{

int data;                     //数据域

struct Node* per;            //前指针域

struct Node* next;          //后指针域三部分

}Node;                //给结构体重新起一个名字                

   

     //创建并初始化

Node*  initList()      //返回的结点类型为Node*

{

Node* L = (Node*)malloc(sizeof(Node));     //动态开辟一个结点

assert(L);        //判断L是否成功开辟,如果开辟失败直接报错

L->data = 0;      //头结点的数据域来储存链表元素个数

L->per = 0;  

L->next = 0;       //两个指针置空

return L;           //返回结点

}


2.   头插法添加结点

void headInsert(Node* L,int x)

{

Node* node = (Node*)malloc(sizeof(node));  //动态开辟一个结点

assert(node);                              //判断L是否成功开辟,如果开辟失败直接报错

node->data = x;              //x是想插入的元素

if (L->data==0)               //判断是否为第一次插入,如果为空进入if语句

{

 node->next = NULL;                   //因为 为空与不为空操作步骤不一样    

 node->per = L;

 L->next = node;

}

else

{                              //不为空,

 node->per = L;

 node->next = L->next;

 L->next->per = node;

 L->next = node;

}

L->data++;         //记录元素个数,加一

}


3.  尾插法添加结点

 

void tailList(Node* L, int x)

{

Node* n = L;

Node* node = (Node*)malloc(sizeof(Node));

assert(node);

node->data = x;          //道理和上面代码类似,本处便不再解释


while (n->next)         //循环遍历,作用:找到最后一个结点

 {

 n = n->next;

 }

                                 //找到会跳出循环,进行尾插

node->next = n->next;         //先连接尾部

n->next = node;

node->per = n;               //连接头部

L->data++;                   //记录长度元素个数加一


}


4.  删除任意位置结点

void deleteList(Node* L,int data)

{

Node* node = L->next;          

if (node == NULL)              //判断链表是否为空

{

 printf("不存在");

        exit(-1);                 //中断程序

}

while (node)                   //判断node结点是否为空指针

{

 if (node->data == data)     //找到后还要分两种情况

 {

  if (node->next == NULL)   //第1种为尾结点

  {

   node->per->next = NULL;

   node->per = NULL;

   break;                //跳出循环

  }

  node->next->per = node->per;  //第2种不为尾结点

  node->per->next = node->next;

 }

 node = node->next;            //循环遍历,当node为空指针时跳出循环

 }

free(node);         //释放要删除的结点

node = NULL;       //避免野指针出现

L->data--;         //记录元素个数减一

}


5.打印链表

   

void printList(Node* L)

{

Node* node = L->next;           //跳过头结点

while (node != NULL)            //循环遍历每一个元素

{

 printf("%d->", node->data);

 node = node->next;

}

printf("NULL\n");           //为了美观

}

6.主函数,来调用其它子函数

int main()

{


Node* L=initList();        

headInsert(L, 1);         //头插添加元素1

headInsert(L, 2);         //头插添加元素2

tailList(L, 1);           //头插添加元素1

tailList(L, 2);           //头插添加元素2

printList(L);       //打印链表

deleteList(L, 2);         //删除元素为2的结点

deleteList(L, 1);         //删除元素为1的结点

printList(L);       //再打印链表

return 0;

}

整体代码

#include<stdio.h>

#include<stdlib.h>

#include<assert.h>

typedef struct Node

{

int data;

struct Node* per;

struct Node* next;

}Node;

        //初始化

Node*  initList()

{

Node* L = (Node*)malloc(sizeof(Node));

assert(L);

L->data = 0;

L->per = 0;

L->next = 0;

return L;

}

       //头插法

void headInsert(Node* L,int x)

{

Node* node = (Node*)malloc(sizeof(node));

assert(node);

node->data = x;

if (L->data==0)

{

 node->next = NULL;

 node->per = L;

 L->next = node;

 L->data++;

}

else

{

 node->per = L;

 node->next = L->next;

 L->next->per = node;

 L->next = node;

}

L->data++;         //记录元素个数

}

      //尾插法

   

void tailList(Node* L, int x)

{

Node* n = L;

Node* node = (Node*)malloc(sizeof(Node));

assert(node);

node->data = x;


while (n->next)

 {

 n = n->next;

 }

node->next = n->next;

n->next = node;

node->per = n;

L->data++;


}

      //删除

   

void deleteList(Node* L,int data)

{

Node* node = L->next;

if (node == NULL)

{

 printf("不存在");

 exit(-1);

}

while (node)

{

 if (node->data == data)

 {

  if (node->next == NULL)

  {

   node->per->next = NULL;

   node->per = NULL;

   break;

  }

  node->next->per = node->per;

  node->per->next = node->next;

 }

 node = node->next;

 }

free(node);

node = NULL;

L->data--;

}

      //打印

void printList(Node* L)

{

Node* node = L->next;

while (node != NULL)

{

 printf("%d->", node->data);

 node = node->next;

}

printf("NULL\n");

}

int main()

{


Node* L=initList();

/*headInsert(L, 1);

headInsert(L, 2);

tailList(L, 1);

tailList(L, 2);

printList(L);

deleteList(L, 2);*/

deleteList(L, 1);

printList(L);

return 0;

}


所有注释都是我亲自写的,如果文章对您有帮助,别吝啬您的手指嘿,嘿嘿嘿,谢谢大家,热烈欢迎大家评论!!


目录
相关文章
|
19天前
|
存储 算法 C语言
【C语言】深入浅出:C语言链表的全面解析
链表是一种重要的基础数据结构,适用于频繁的插入和删除操作。通过本篇详细讲解了单链表、双向链表和循环链表的概念和实现,以及各类常用操作的示例代码。掌握链表的使用对于理解更复杂的数据结构和算法具有重要意义。
144 6
|
23天前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
44 5
|
1月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
83 4
|
6月前
|
存储 缓存 前端开发
【数据结构/C语言】深入理解 双向链表
【数据结构/C语言】深入理解 双向链表
|
2月前
|
存储 缓存 C语言
C语言:链表和数组有什么区别
C语言中,链表和数组是两种常用的数据结构。数组是一种线性结构,元素在内存中连续存储,通过下标访问,适合随机访问且大小固定的情况。链表由一系列不连续的节点组成,每个节点存储数据和指向下一个节点的指针,适用于频繁插入和删除操作的场景,链表的大小可以动态变化。
|
2月前
|
C语言
无头链表再封装方式实现 (C语言描述)
如何在C语言中实现无头链表的再封装,包括创建节点和链表、插入和删除操作、查找和打印链表以及销毁链表的函数。
29 0
|
2月前
|
C语言
C语言链式结构之有头单链表再封装写法
本文介绍了如何使用C语言对有头单链表进行封装,包括节点的创建、链表的初始化、数据的插入和删除,以及链表的打印等功能。
21 1
|
2月前
|
C语言
C语言结构体链式结构之有头单链表
文章提供了一个C语言实现的有头单链表的完整代码,包括创建链表、插入、删除和打印等基本操作。
35 1
|
1月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
56 0
|
2月前
|
C语言
无头链表二级指针方式实现(C语言描述)
本文介绍了如何在C语言中使用二级指针实现无头链表,并提供了创建节点、插入、删除、查找、销毁链表等操作的函数实现,以及一个示例程序来演示这些操作。
36 0