【数据结构】无头+单向+非循环链表(SList)(增、删、查、改)详解

简介: 【数据结构】无头+单向+非循环链表(SList)(增、删、查、改)详解

一、链表的概念及结构

1、链表的概念

之前学习的顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,而链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的,可以实现更加灵活的动态内存管理。


:之所以引出链表,是因为顺序表存在一些缺点

  • 顺序表在中间 / 头部的插入和删除,需要挪动很多数据,时间复杂度为 O(N),效率低下。
  • 增容需要申请新空间,拷贝数据,释放旧空间。消耗不小。
  • 增容一般是一次增长 2 倍,那就一定会造成空间浪费。例如当前的容量为 100,满了以后增容到 200,这时再继续插入 5 个数据,后面不再插入,那么就浪费了 95 个数据空间。

2、链表的组成

链表由一系列结点(链表中每一个元素称为结点)组成,结点可以在运行时动态生成

每个结点包括两个部分:

  1. 数据域:存储数据元素。
  2. 指针域:存储下一个结点地址。

3、链表的结构

(1)链表的物理结构(现实中)

(2)链表的逻辑结构(想象中)


  • 链式结构在逻辑上是连续的,但在物理上不一定连续。
  • 现实中的结点一般都是上申请出来的
  • 从堆上申请的空间,是按照一定的策略来分配的,两次申请的空间可能连续,也可能不连续。

二、无头+单向+非循环链表的接口实现

无头单向非循环链表结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。


1、创建文件

  • test.c(主函数、测试顺序表各个接口功能)
  • SList.c(单链表接口函数的实现)
  • SList.h(单链表的类型定义、接口函数声明、引用的头文件)


2、SList.h 头文件代码

// SList.h
// 无头+单向+非循环链表增删查改实现
#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
 
typedef int SLTDateType;
 
typedef struct SListNode
{
    SLTDateType data; // 数据域
    struct SListNode* next; // 指针域
}SListNode;
 
// 动态申请一个节点
SListNode* BuySListNode(SLTDateType x);
// 销毁单链表中所有节点
void SListDestory(SListNode** pphead)
// 单链表打印
void SListPrint(SListNode* phead);
// 单链表尾插
void SListPushBack(SListNode** pphead, SLTDateType x);
// 单链表的头插
void SListPushFront(SListNode** pphead, SLTDateType x);
// 单链表的尾删
void SListPopBack(SListNode** pphead);
// 单链表头删
void SListPopFront(SListNode** pphead);
// 单链表查找
SListNode* SListFind(SListNode* phead, SLTDateType x);
// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x);
// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos);
// 求单链表长度
int SListSize(SListNode* phead);
// 单链表判空
bool SListEmpty(SListNode* phead);

三、在 SList.c 中实现各个接口函数

1、动态申请一个节点

// 动态申请一个节点
SListNode* BuyListNode(SLTDataType x)
{
  SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
  newnode->data = x;
  newnode->next = NULL;
  return newnode;
}

2、销毁(释放)所有节点

// 销毁单链表中所有节点
void SListDestory(SListNode** pphead)
{
    assert(pphead);
  SListNode* cur = *pphead;
  while (cur)
  {
    SListNode* next = cur->next;
    free(cur); // 释放节点
    cur = next;
  }
  *pphead = NULL;
}

assert() 放在函数里面检查参数,一方面是为了安全,另一方面是为了防止其他人在调用该函数时,出现不正确的使用,导致错误传入参数,这样可以及时提醒到他。所以写代码时一定要考虑到其他人不正确的使用该函数时的场景,以此避免不必要的麻烦。


3、单链表打印

// 单链表打印
void SListPrint(SListNode* phead)
{
    // 不需要断言 -- 空链表也可以打印
    SListNode* cur = phead;
  while (cur)
  {
    printf("%d->", cur->data);
    cur = cur->next;
  }
  printf("NULL\n");
}

4、单链表尾插

// 单链表尾插
void SListPushBack(SListNode** pphead, SLTDateType x)
{
  SListNode* newnode = BuySListNode(x);  //动态申请一个节点
 
  if (*pphead == NULL) // 单链表中没有节点时
  {
    *pphead = newnode; // plist指向新节点
  }
 
  else // 单链表中已经有节点时
  {
    SListNode* tail = *pphead;
    while (tail->next != NULL) // 找到单链表中的最后一个节点
    {
      tail = tail->next;
    }
    tail->next = newnode; // 最后一个节点的next指向新节点
  }
}

 错误写法❌:

(传一级指针的值,用一级指针接收)指针传值,相当于把 plist 指针变量的值拷贝给 phead,给 phead 赋值 newnode,phead 的改变不会影响 plist

// 错误写法:
// 单链表尾插
void SListPushBack(SListNode* phead, SLTDateType x)
{
  SListNode* newnode = BuySListNode(x);  //动态申请一个节点
 
  if (phead == NULL) // 单链表中没有节点时
  {
    phead = newnode; // plist指向新节点
  }
 
  else // 单链表中已经有节点时
  {
    SListNode* tail = phead; // 找到尾节点
    while (tail->next != NULL) // 找到单链表中的最后一个节点
    {
      tail = tail->next;
    }
    tail->next = newnode; // 最后一个节点的next指向新节点
  }
}

因为当链表为空时,我们需要改变 plist 的指向,使其指向第一个节点。而初始 plistphead 都指向 NULL,调用函数后,phead 指向了新的节点,而 plist 还是指向 NULL 的。


解决方法:

plist 是指向第一个节点的指针,想要在函数中改变 plist 的值(指向),必须要把 plist 指针的地址 作为实参传过去,形参用二级指针接收,这样在函数中对二级指针解引用得到 plist 的值,就可以改变 plist 的值(指向)了。


  • 单链表为空时,plist 直接指向新节点
  • 单链表不为空时,先找到单链表的尾节点,然后将尾节点的 next 指向新节点


5、单链表头插

// 单链表的头插
void SListPushFront(SListNode** pphead, SLTDataType x)
{
  assert(pphead);
  SListNode* newnode = BuyListNode(x); // 动态申请一个节点
  newnode->next = *pphead; // 新节点的next指针指向plist指向的位置
  *pphead = newnode; // plist指向头插的新节点
}


6、单链表尾删

// 单链表的尾删
void SListPopBack(SListNode** pphead)
{
  assert(pphead);
    assert(*pphead); //链表为空就无法再进行尾删了
 
  // 温柔处理方式
  /*if (*pphead == NULL)
  {
    return;
  }*/
 
  // 粗暴处理方式
  assert(*pphead);
 
  if ((*pphead)->next == NULL) // 链表一个节点
  {
    free(*pphead);
    *pphead = NULL;
  }
 
  else // 链表中有多个节点
  {
        // 写法一:
    /* SListNode* prev = NULL;
    SListNode* tail = *pphead;
    while(tail->next != NULL) // 找到链表的尾节点的上一个节点
    {
      prev = tail;
      tail = tail->next;
    }
    free(tail); // 删除尾节点
    tail = NULL;
    prev->next = NULL;  // 置空 */
 
        //写法二:
    SListNode* tail = *pphead;
    while (tail->next->next != NULL) // 找到链表的尾节点的上一个节点
    {
      tail = tail->next;
    }
    free(tail->next); // 删除尾节点
    tail->next = NULL; // 置空
  }
}


  • 单链表只有一个节点时,删除节点plist 指向 NULL
  • 单链表有多个节点时,先找到单链表尾节点的上一个节点删除尾节点,然后将该节点的 next 指向 NULL
  • 因为可能要改变外部 plist 的指向,所以用二级指针接收指针 plist 的地址

7、单链表头删

// 单链表头删
void SListPopFront(SListNode** pphead)
{
  assert(pphead);
    assert(*pphead); // 链表为空就无法再进行头删了
 
  /*if (*pphead == NULL)
  {
    return;
  }
  else
  {
    SListNode* next = (*pphead)->next;
    free(*pphead);
    *pphead = next;
  }*/
 
  SListNode* cur = *pphead; // 保存头节点的地址
  *pphead = cur->next; // plist指向头节点的下一个节点
  free(cur); // 删除头节点
}

:因为可能要改变外部 plist 的指向,所以用二级指针接收指针 plist 的地址


8、单链表查找指定值的节点

// 单链表查找
SLTNode* SListFind(SListNode* phead, SLTDataType x)
{
  SListNode* cur = phead;
  while (cur)
  {
    if (cur->data == x)
    {
      return cur; // 找到了 返回该节点的地址
    }
    else
    {
      cur = cur->next;
    }
  }
  return NULL; // 未找到,返回NULL
}


9、单链表在pos位置之后插入x

// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x)
{
  assert(pos); // pos位置不能为空
  SListNode* newnode = BuySListNode(x); // 动态申请一个节点
  newnode->next = pos->next; // 新节点的next指针指向pos位置后一个节点
  pos->next = newnode; // pos位置的next指向新节点
}

为什么不在pos位置之前插入?
  • 单链表不适合在 pos 位置之前插入,因为需要遍历链表找到 pos 位置的前一个节点,时间复杂度为O(N)。
  • 单链表更适合在 pos 位置之后插入,如果在后面插入,只需要知道 pos 位置即可,简单得多。
  • C++ 官方库里面单链表给的也是在之后插入


10、单链表删除指定pos位置之后的节点

// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos)
{
  assert(pos); // pos位置不能为空
  assert(pos->next); // pos位置不能是尾节点
  SListNode* next = pos->next; // 保存pos位置的后一个节点
  pos->next = pos->next->next;
  free(next); // 释放pos位置的后一个节点
}

为什么不删除pos位置?
void SListErase(SListNode** pphead, SLTNode* pos) // O(N)
{
  assert(pphead);
  assert(pos);
 
  if (*pphead == pos)
  {
    /* *pphead = pos->next;
    free(pos); */
    SListPopFront(pphead);
  }
  else
  {
    SListNode* prev = *pphead;
    while (prev->next != pos)
    {
      prev = prev->next;
    }
    prev->next = pos->next;
    free(pos);
    //pos = NULL;
  }
}
  • 删除 pos 位置同样需要传入单链表的二级指针。
  • 单链表不适合在 pos 位置插入,因为需要遍历链表找到 pos 位置的前一个节点,以改变其指向,时间复杂度大。


11、求单链表长度

// 求单链表长度
int SListSize(SListNode* phead)
{
  int size = 0;
  SListNode* cur = phead;
  while (cur)
  {
    size++;
    cur = cur->next;
  }
  return size;
}

12、判断单链表是否为空

// 单链表判空
bool SListEmpty(SListNode* phead)
{
    // 写法一:
  return phead == NULL;
  
  // 写法二:
  //return phead == NULL ? true : false;
}

四、代码整合

// SList.c
#include "SList.h"
 
// 动态申请一个节点
SListNode* BuyListNode(SLTDataType x)
{
  SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
  newnode->data = x;
  newnode->next = NULL;
  return newnode;
}
 
// 销毁单链表中所有节点
void SListDestory(SListNode** pphead)
{
    assert(pphead);
  SListNode* cur = *pphead;
  while (cur)
  {
    SListNode* next = cur->next;
    free(cur); // 释放节点
    cur = next;
  }
  *pphead = NULL;
}
 
// 单链表打印
void SListPrint(SListNode* phead)
{
    SListNode* cur = phead;
  while (cur)
  {
    printf("%d->", cur->data);
    cur = cur->next;
  }
  printf("NULL\n");
}
 
// 单链表尾插
void SListPushBack(SListNode** pphead, SLTDateType x)
{
  SListNode* newnode = BuySListNode(x);  //动态申请一个节点
 
  if (*pphead == NULL) // 单链表中没有节点时
  {
    *pphead = newnode; // plist指向新节点
  }
 
  else // 单链表中已经有节点时
  {
    SListNode* tail = *pphead;
    while (tail->next != NULL) // 找到单链表中的最后一个节点
    {
      tail = tail->next;
    }
    tail->next = newnode; // 最后一个节点的next指向新节点
  }
}
 
// 单链表的头插
void SListPushFront(SListNode** pphead, SLTDataType x)
{
  assert(pphead);
  SListNode* newnode = BuyListNode(x); // 动态申请一个节点
  newnode->next = *pphead; // 新节点的next指针指向plist指向的位置
  *pphead = newnode; // plist指向头插的新节点
}
 
// 单链表的尾删
void SListPopBack(SListNode** pphead)
{
  assert(pphead);
    assert(*pphead); //链表为空就无法再进行尾删了
  assert(*pphead);
 
  if ((*pphead)->next == NULL) // 链表一个节点
  {
    free(*pphead);
    *pphead = NULL;
  }
  else // 链表中有多个节点
  {
    SListNode* tail = *pphead;
    while (tail->next->next != NULL) // 找到链表的尾节点的上一个节点
    {
      tail = tail->next;
    }
    free(tail->next); // 删除尾节点
    tail->next = NULL; // 置空
  }
}
 
// 单链表头删
void SListPopFront(SListNode** pphead)
{
  assert(pphead);
    assert(*pphead); // 链表为空就无法再进行头删了
 
  SListNode* cur = *pphead; // 保存头节点的地址
  *pphead = cur->next; // plist指向头节点的下一个节点
  free(cur); // 删除头节点
}
 
// 单链表查找
SLTNode* SListFind(SListNode* phead, SLTDataType x)
{
  SListNode* cur = phead;
  while (cur)
  {
    if (cur->data == x)
    {
      return cur; // 找到了 返回该节点的地址
    }
    else
    {
      cur = cur->next;
    }
  }
  return NULL; // 未找到,返回NULL
}
 
// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x)
{
  assert(pos); // pos位置不能为空
  SListNode* newnode = BuySListNode(x); // 动态申请一个节点
  newnode->next = pos->next; // 新节点的next指针指向pos位置后一个节点
  pos->next = newnode; // pos位置的next指向新节点
}
 
// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos)
{
  assert(pos); // pos位置不能为空
  assert(pos->next); // pos位置不能是尾节点
  SListNode* next = pos->next; // 保存pos位置的后一个节点
  pos->next = pos->next->next;
  free(next); // 释放pos位置的后一个节点
}
 
// 求单链表长度
int SListSize(SListNode* phead)
{
  int size = 0;
  SListNode* cur = phead;
  while (cur)
  {
    size++;
    cur = cur->next;
  }
  return size;
}
 
// 单链表判空
bool SListEmpty(SListNode* phead)
{
    // 写法一:
  return phead == NULL;
}


相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
65 4
|
13天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
75 5
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
122 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
57 0
|
3月前
|
存储 Java
数据结构第三篇【链表的相关知识点一及在线OJ习题】
数据结构第三篇【链表的相关知识点一及在线OJ习题】
37 7
|
3月前
|
存储 安全 Java
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
31 3
|
3月前
|
算法 Java
数据结构与算法学习五:双链表的增、删、改、查
双链表的增、删、改、查操作及其Java实现,并通过实例演示了双向链表的优势和应用。
30 0
数据结构与算法学习五:双链表的增、删、改、查

热门文章

最新文章