深入剖析无头单链表——C语言动图演示

简介: 深入剖析无头单链表——C语言动图演示

🔊个人介绍

🌳数据结构系列点击跳转

🌳戳我跳到本人个人主页

🇨🇳大家好,我是_奇奇,一名C/C++博主。河牧院大一在读。

🔔欢迎大家交流学习

❤️编程的前途是光明的,道路是曲折的。

🌳笑到最后才是赢家🍺


这里是目录

🌳无头单向链表

🌳结构体声明(节点)

🌳动态申请新节点

🌳尾插

1.当链表为空的时候。

2.当链表不为空的时候。

🌳打印

🌳头插

1.当表为空时

2.当表不为空时

🌳尾删

1.当表不为空时

2.当表为空时

3.当表只有一个节点时

🌳头删

1.当表为空时

2.当表为一个节点时

3.当表不为空时

🌳查找

🌳在指定元素前面插入一个节点

1.当头指针等于指定元素的地址时

2.一般情况

🌳销毁

1.当表为空时

2.当表不为空时

🌳test.c

🌳SList.h

🌳SList.c


单链表的实现,学习单链表就需要对C语言指针熟悉了解。因为链表,顾名思义就是用链子连起来的的表。而这个链子就是指针。实际上就是地址。通过地址来连接起来的表。

画图画图画图,让我们来看看单链表是什么样子吧。

我实现的这个单链表是没有头的。不带首元节点。只有一个头指针。指向第一个元素。单链表的每个元素(一般称之为节点)都是一个结构体类型。每个这样的节点都包含一个指向下一个节点的指针和一个数据。

如下图所示。

假设每个节点的地址都是已知的。我把每个结构体的地址都写了出来


🌳无头单向链表


和严老师书上的不同的是,我们这个单链表没有首元结点,原因是刚开学单链表先从简单开始,否则容易迷糊。

链表的实质就是下面的一幅图。这张图只要看懂了。单链表你就会了。

链表的实质是通过16进制表示的地址连接起来的,可以这么说地址就是指针如图所示。

指针相当于绿色的箭头。头指针plist里面存放的是第一个节点的地址0x00000030,第一个节点里面的next指针里面存放的是第二个节点的地址0x00000050,第一个节点里面的next指针里面存放的是第二个节点的地址0x00000010…以此类推直到最后一个节点的next指针指向NULL;


🌳结构体声明(节点)

在头文件中对结构体进行声明创建,为什么起名叫SListNode,因为是Signle List Node (单链表)的缩写;

将int类型typedef为SLNDataType的原因是易于后期更换每个节点的数据。

每个SListNode节点包含一个int类型数据 和 一个指向这个节点的指针next。


🌳动态申请新节点

  • 在尾插之前需先申请一个节点,我们把它命名为newnode。按需所取。不浪费空间。
  • 再申请的同时我们需要这个新节点初始化了。
  • 把next指针赋值为空指针。数据传我们想传递的数据。


  • 如果开辟失败了,就结束程序。
  • else将节点初始化。
  • 最后返回节点。

头文件中函数声明

源文件中函数定义


🌳尾插

在这里有点需要注意。那就是二级指针。因为需要修改头指针里面的地址所以要传头指针的地址。传地址才能修改里面的值。


原因是因为形参只是实参的一份临时拷贝。只有传地址才能改变里面的值。


在严蔚敏老师的数据结构书中用的是C++语法中的引用,是&符号,和传地址的效果一样。


尾插之前先调用申请空间的函数。



  • 在尾部插入数据分为两种情况

1.当链表为空的时候。

为空时直接将newnode这个地址赋给*pphead就ok了。


2.当链表不为空的时候。

需要先找到尾结点。这时我们定义一个指向头节点的指针tail

1.



tail->next不为空,就执行tail = tail->next.意思就是让tail指针向前移动一个节点



3.不为空,向前移动一个节点



4.newnode->next是空。跳出while循环


5.加入新节点



6.将newnode的值赋给tail->next.就此完成了尾删。



🌳打印

  • 打印依次遍历链表就ok了。
    源文件test源文件



🌳头插

1.当表为空时

两种情况可以合二为一。

2.当表不为空时

1.在第一个节点处插入

2.创建新节点

3.第一步将newnode->next指针里面存放第一个节点的地址0x0000 0030

第二步再将newnode的值0x00000020赋给头指针。

顺序不能错,否则逻辑错误。

这样就完成了头插的解释。


🌳尾删

二话不说,先画图。


1.当表不为空时

定义一个prev指针变量。用来找到倒数第二个节点并把他的next指针赋为NULL。当tail->next不为空,prev指向tail指向的位置。tail向前移动。

两个指针继续移动直至tail->next 为NULL。当tail->next为空时跳出循环

跳出循环然后prev->next置为空。free掉tail的空间。tail置为空


2.当表为空时

当表为空时直接return。

3.当表只有一个节点时

直接free掉 头指针(*pphead)指向的空间。

记得给头指针赋值为空。否则头指针就是野指针了

尾删的解释到此结束


🌳头删

1.当表为空时


2.当表为一个节点时

只有一个节点和有多个节点代码一样。可以合二为一。

3.当表不为空时

先定一个current变量用来记住第二个节点的地址。(因为free掉第一个节点后。第二个节点的地址就无法访问了。)然后再free掉头指针。再把current里存放的值放给头指针。



🌳查找

查找某个元素,若找到返回他的地址。否则返回空指针。


🌳在指定元素前面插入一个节点

先调用查找函数。找到这个元素的地址。然后返回。

1.当头指针等于指定元素的地址时

这个情况和头插一样。看前面的头插


2.一般情况

和尾插情况一样。详情见尾插。


🌳销毁

为什么要销毁?

当我们的链表还有节点 或者 不想继续用下去的时候可以销毁掉。


1.当表为空时

此时不进入while循环,直接将头指针赋为空。

2.当表不为空时

设置一个临时变量tmp。用来保存头指针cur的next里面地址。

然后free掉头指针指向的空间。

再让头指针前进一个节点,直至遍历完整个链表。

此时就结束了整个链表。


各位观众老爷,创作不易,喜欢的话来个三连吧、

🌳test.c

#define _CRT_SECURE_NO_WARNINGS
#include "SList.h"
int main()
{
  SListNode* plist = NULL;
  SListNodePushBack(&plist, 1);
  SListNodePushBack(&plist, 2);
  SListNodePushBack(&plist, 3);
  SListNodePushBack(&plist, 4);
  SListNodePushBack(&plist, 1);
  SListNodePushFront(&plist, 0);
  SListNodePushFront(&plist, -1);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePopBack(&plist);
  SListNodePushFront(&plist, 0);
  SListNodePushFront(&plist, -1);
  SListNode* pos = SListNodeFind(&plist, 0);
  SListInsert(&plist, pos, 6);
  SListNodePrint(&plist);
  SListNodeDestory(&plist);
  SListNodePrint(&plist);
  return 0;
}

🌳SList.h

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
typedef int SLNDataType;
typedef struct SListNode
{
  SLNDataType data;
  struct SListNode* next;
}SListNode;
//动态申请一个节点
SListNode* SListNodeBuyNewnode(SLNDataType x);
//尾插
void SListNodePushBack(SListNode** pphead, SLNDataType x);
//打印
void SListNodePrint(SListNode** pphead);
//头插
void SListNodePushFront(SListNode** pphead, SLNDataType x);
//尾删
void SListNodePopBack(SListNode** pphead);
//头删
void SListNodePopFront(SListNode** pphead);
//查找
SListNode* SListNodeFind(SListNode** pphead, SLNDataType x);
//在一个地址前面插入一个节点
void SListInsert(SListNode** pphead, SListNode* pos, SLNDataType x);
//销毁
void SListNodeDestory(SListNode** pphead);

🌳SList.c

#define _CRT_SECURE_NO_WARNINGS
#include "SList.h"
//动态申请一个节点
SListNode* SListNodeBuyNewnode(SLNDataType x)
{
  SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
  if (NULL == newnode)
  {
    printf("malloc fail");
    exit(-1);
  }
  else
  {
    newnode->data = x;
    newnode->next = NULL;
  }
  return newnode;
}
//尾插
void SListNodePushBack(SListNode** pphead, SLNDataType x)
{
  SListNode* newnode = SListNodeBuyNewnode(x);
  //当链表为空的时候
  if (*pphead == NULL)
  {
    *pphead = newnode;
  }
  //当不为空的时候
  else
  {
    SListNode* tail = *pphead;
    while (tail->next != NULL)
    {
      tail = tail->next;
    }
    tail->next = newnode;
  }
}
//打印
void SListNodePrint(SListNode** pphead)
{
  while (*pphead != NULL)
  {
    printf("%d->", (*pphead)->data);
    (*pphead) = (*pphead)->next;
  }
  printf("NULL\n");
}
//头插
void SListNodePushFront(SListNode** pphead, SLNDataType x)
{
  SListNode* newnode = SListNodeBuyNewnode(x);
  newnode->next = *pphead;
  *pphead = newnode;
}
//尾删
void SListNodePopBack(SListNode** pphead)
{
  assert(pphead);
  //当表为空时
  if (*pphead == NULL)
  {
    return;
  }
  //当表只有一个节点
  else if ((*pphead)->next == NULL)
  {
    free(*pphead);
    *pphead = NULL;
    return;
  }
  //当表不为空
  else 
  {
    SListNode* prev = NULL;
    SListNode* tail = *pphead;
    while (tail->next != NULL)
    {
      prev = tail;
      tail = tail->next;
    }
    prev->next = NULL;
    free(tail);
    tail = NULL;
  }
}
//头删
void SListNodePopFront(SListNode** pphead)
{
  assert(pphead);
  if (*pphead == NULL)
  {
    return;
  }
  SListNode* current = (*pphead)->next;
  free(*pphead);
  *pphead = current;
}
//查找
SListNode* SListNodeFind(SListNode** pphead, SLNDataType x)
{
  SListNode* cur = *pphead;
  while (cur)
  {
    if (cur->data == x)
    {
      return cur;
    }
    else
    {
      cur = cur->next;
    }
  }
  return NULL;
}
//在一个地址前面插入一个节点
void SListInsert(SListNode** pphead, SListNode* pos, SLNDataType x)
{
  SListNode* newnode = SListNodeBuyNewnode(x);
  if (*pphead == pos)
  {
    newnode->next = *pphead;
    *pphead = newnode;
  }
  else
  {
    SListNode* prev = *pphead;
    while (prev->next != pos)
    {
      prev = prev->next;
    }
    prev->next = newnode;
    newnode->next = pos;
  }
}
//销毁
void SListNodeDestory(SListNode** pphead)
{
  assert(pphead);
  SListNode* cur = *pphead;
  while (cur)
  {
    SListNode* tmp = cur->next;
    free(cur);
    cur = tmp;
  }
  *pphead = NULL;
}
相关文章
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
119 4
|
3月前
|
C语言
无头链表再封装方式实现 (C语言描述)
如何在C语言中实现无头链表的再封装,包括创建节点和链表、插入和删除操作、查找和打印链表以及销毁链表的函数。
37 0
|
3月前
|
测试技术 C语言
单链表之无头链表(C语言版)
本文详细介绍了使用C语言实现无头单链表的方法,包括节点和链表结构的定义、链表的创建与销毁、节点的插入与删除,以及链表的打印等功能。文章通过具体的代码示例,展示了如何在无头链表中进行头插法、尾插法、自定义位置插入和删除,以及如何清空和销毁链表。
53 0
单链表之无头链表(C语言版)
|
3月前
|
存储 C语言
C语言单链表实现
一个用C语言编写的简单学生信息管理系统,该系统具备信息输入、成绩计算、排序、删除、查找、修改、保存和读取文件等功能。
33 0
C语言单链表实现
|
2月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
79 0
|
3月前
|
C语言
无头链表二级指针方式实现(C语言描述)
本文介绍了如何在C语言中使用二级指针实现无头链表,并提供了创建节点、插入、删除、查找、销毁链表等操作的函数实现,以及一个示例程序来演示这些操作。
42 0
|
4月前
|
存储 算法 C语言
数据结构基础详解(C语言):单链表_定义_初始化_插入_删除_查找_建立操作_纯c语言代码注释讲解
本文详细介绍了单链表的理论知识,涵盖单链表的定义、优点与缺点,并通过示例代码讲解了单链表的初始化、插入、删除、查找等核心操作。文中还具体分析了按位序插入、指定节点前后插入、按位序删除及按值查找等算法实现,并提供了尾插法和头插法建立单链表的方法,帮助读者深入理解单链表的基本原理与应用技巧。
740 6
|
4月前
|
存储 算法 C语言
C语言手撕实战代码_循环单链表和循环双链表
本文档详细介绍了用C语言实现循环单链表和循环双链表的相关算法。包括循环单链表的建立、逆转、左移、拆分及合并等操作;以及双链表的建立、遍历、排序和循环双链表的重组。通过具体示例和代码片段,展示了每种算法的实现思路与步骤,帮助读者深入理解并掌握这些数据结构的基本操作方法。
|
4月前
|
算法 C语言 开发者
C语言手撕实战代码_单链表
本文档详细介绍了使用C语言实现单链表的各种基本操作和经典算法。内容涵盖单链表的构建、插入、查找、合并及特殊操作,如头插法和尾插法构建单链表、插入元素、查找倒数第m个节点、合并两个有序链表等。每部分均配有详细的代码示例和注释,帮助读者更好地理解和掌握单链表的编程技巧。此外,还提供了判断子链、查找公共后缀等进阶题目,适合初学者和有一定基础的开发者学习参考。
|
6月前
|
存储 数据管理 C语言
C语言实战 | 使用链表完成“贪吃蛇”游戏
【7月更文挑战第1天】整体思维,即系统思维,强调以整体视角理解事物。在编程中,结构体体现这种思想,将相关变量打包处理。示例展示了如何用链表而非数组实现“贪吃蛇”游戏,链表提供了更灵活的动态数据管理。一系列代码图片详细描绘了链表结构体在游戏中的应用,包括节点定义、移动、碰撞检测等,凸显了使用链表的优势和代码的清晰组织。
72 0
C语言实战 | 使用链表完成“贪吃蛇”游戏