数据结构之单链表详解(C语言手撕)

简介: 数据结构之单链表详解(C语言手撕)


一.链表的概念及结构

概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的

从图片中可以看出,链表的每个节点都是一个结构体,该结构体中有一个存储数据的变量和一个指向下一节点的结构体指针;
在逻辑上是连续的,但是在物理空间上不一定连续,因为链表节点都是每次插入数据时在堆上申请出来的;每次申请出来的空间不一定是连续的;

二.无头单向非循环链表的结构

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

二.无头单向非循环链表的结构及实现

一.结构:

二.功能函数的实现(含性能分析与图解)
  1. 打印链表
  2. 创建节点函数
  3. 头插节点函数
  4. 头删节点函数
  5. 尾插节点函数
  6. 尾删节点函数
  7. 查找函数
  8. 在一节点位置之前插入一个节点
  9. 在一节点位置之后插入一个节点
  10. 在一节点位置之前删除一个节点
  11. 在一节点位置之后删除一个节点
打印链表

图解:遍历访问

先定义一个结点指针指向头节点,往后依次遍历,与数组不同的是,不是cur++,而是让cur指向下一个节点,即cur=cur->next;

代码实现:

void SLPrint(SLNode* pphead)
{
  assert(pphead);     //断言
  SLNode* cur = pphead;     //让cur指向头节点进行遍历
  while (cur)        //注意:条件不是cur->next,因为如果是cur->next为空就不进入循环的话,则最后一个节点就访问不到
  {
    printf("%d ", cur->val);
    cur = cur->next;
  }
  printf("NULL");    //最后打印一个NULL、方便观察
  printf("\n");
}

性能分析:

时间复杂度:O(N)

空间复杂度:O(1)

创建节点函数

代码实现:

SLNode* BuySLnewnode(SLDateType x)
{
  SLNode* newnode = (SLNode*)malloc(sizeof(SLNode));   //注意:创建的是节点,一个结构体,而不是一个数据类型
  if (newnode == NULL)          //判断
  {
    perror("malloc fail");
    exit(-1);             //开辟失败,以异常退出程序
  }
  newnode->next = NULL;     //下一个节点置NULL
  newnode->val = x;         //赋值
  return newnode;           //返回该该结点指针
}
头插节函数

图解:

代码实现:

void SLPushFront(SLNode** pphead, SLDateType x)   
//注意:这里需要节点的二级指针,因为外面调用的时候,如果传的是头指针的话,是传值,
//函数里面改变不影响头指针的指向,所以这里需要二级指针,函数调用的时候需要传二级指针
{
  SLNode* newnode = BuySLnewnode(x);     //创建新节点
  if (*pphead == NULL)       //检查,如果为空链表
  {
    *pphead = newnode;      //直接将*pphead指向新节点
  }
  else
  {
    newnode->next = *pphead;    //第二种情况
    (*pphead) = newnode;
  }
}

性能分析:

时间复杂度:O(1)

空间复杂度:O(1)

头删节点函数

图解:

代码实现:

void SLPopFront(SLNode** pphead)
{
  assert(*pphead);    //头指针不能为空
  if((*pphead)->next==NULL)     //第一种情况
  {
    free(*pphead);     
    *pphead = NULL;
    return;
  }
  SLNode* tmp = (*pphead)->next;   //保存下一个节点
  free(*pphead);
  *pphead = tmp;
}

性能分析:

时间复杂度:O(1)

空间复杂度:O(1)

尾插节点函数

图解:

代码实现:

void SLPushBack(SLNode** pphead, SLDateType x)
{
  SLNode* newnode = BuySLnewnode(x);
  if (*pphead == NULL)     //空链表
  { 
    *pphead = newnode;
    return;
  }
  SLNode* tail = *pphead;     //定义一个尾指针
  while (tail->next)
  {
    tail = tail->next;
  }                           //退出循环后tail->next为NULL;
  tail->next = newnode;       //链接
}

性能分析:

时间复杂度:O(N)

空间复杂度:O(1)

尾删节点函数

图解:

代码实现:

在这里插入代码片void SLPopBack(SLNode** pphead)
{
  assert(*pphead);
  if ((*pphead)->next == NULL)   //第一种情况
  {
    free(*pphead);
    *pphead = NULL;
    return;
  }
  SLNode* Prevtail = *pphead;    //记录尾指针前面的一个节点
  SLNode* tail = *pphead;        //尾指针
  while (tail->next)
  {
    Prevtail = tail;           
    tail = tail->next;
  }
  free(tail);             //释放掉尾节点
  Prevtail->next = NULL;   
}

性能分析:

时间复杂度:O(N)

空间复杂度:O(1)

查找函数

代码实现:

SLNode* SLFind(SLNode* pphead, SLDateType x)
{
  assert(pphead);
  SLNode* cur = pphead;       //遍历查找
  while (cur)
  {
    if (cur->val == x)
    {
      return cur;      //返回节点指针
    } 
    cur = cur->next;
  }
  return NULL;         //没找到,返回NULL
}

性能分析:

时间复杂度:O(N)

空间复杂度:O(1)

在pos位置之前插入一个节点

图解:

代码实现:

//在pos之前插入
void SLInsert(SLNode** pphead, SLNode* pos, SLDateType x)
{
  assert(*pphead);
  assert(pos);
  if (pos == *pphead)        //第一种情况:头插
  {
    SLPushFront(pphead, x);
      
    return;
  }
  SLNode* newnode = BuySLnewnode(x);     
  
  SLNode* cur = *pphead;     //遍历,找到pos的前一个节点
  while (cur->next)
  {
    if (cur->next == pos)      //找到了
    {
      newnode->next = cur->next;    //链接
      cur->next = newnode;
      return;
    }
    cur = cur->next;
  }
}

性能分析:

时间复杂度:O(N)

空间复杂度:O(1)

在pos位置之后插入一个节点

图解:

代码实现:

//在pos之后插入
void SLInsertBack(SLNode* pos, SLDateType x)
{
  assert(pos);
  SLNode * newnode = BuySLnewnode(x);     
  newnode->next = pos->next;   //链接
  pos->next = newnode;
}

性能分析:

删除pos位置之前一个节点

图解:

代码实现:

//删除pos之前的节点
void SLErase(SLNode** pphead, SLNode* pos)
{
  assert(pos);
  assert(pos != *pphead);
  if (pos== (*pphead)->next)  //头删,第一种情况
  {
    free(*pphead);
    (*pphead) = pos;
    return;
  }
  SLNode* cur = *pphead;
  while (cur)
  {
    if (cur->next->next == pos)   //找到pos前面的第二个节点
    {
      free(cur->next);
      cur->next = pos;     //链接
      return;
    }
    cur = cur->next;
  }
}

性能分析:

时间复杂度:O(N)

空间复杂度:O(1)

删除pos位置之后一个节点

图解:

代码实现:

//删除pos之后的节点
void SLEraseAfter(SLNode* pos)
{
  assert(pos);
  assert(pos->next);   //当pos后无节点,无意义
  if (pos->next->next == NULL)   //尾删
  {
    pos->next = NULL;
    return;
  }
  SLNode* cur = pos->next->next;   
  free(pos->next);   
  pos->next = cur;   //链接
  cur = NULL;
}

性能分析:

时间复杂度:O(1)

空间复杂度:O(1)

二.总代码
```cpp
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
typedef int SLDateType;
typedef struct SListNode
{
  SLDateType val;
  struct SListNode* next;
}SLNode;
SLNode* BuySLnewnode(SLDateType x);
void SLPrint(SLNode* pphead);
void SLPushBack(SLNode** pphead, SLDateType x);
void SLPushFront(SLNode** pphead, SLDateType x);
void SLPopFront(SLNode** pphead); 
void SLPopBack(SLNode** pphead);
SLNode* SLFind(SLNode* pphead,SLDateType x);
//在pos之前插入
void SLInsert(SLNode** pphead, SLNode* pos,SLDateType x);
//在pos之后插入
void SLInsertBack(SLNode* pos, SLDateType x);
//删除pos之后的节点
void SLEraseBack(SLNode* pos);
//删除pos之前的节点
void SLErase(SLNode** pphead,SLNode* pos);
```cpp
#include"SList.h"
SLNode* BuySLnewnode(SLDateType x)
{
  SLNode* newnode = (SLNode*)malloc(sizeof(SLNode));
  if (newnode == NULL)
  {
    perror("malloc fail");
    exit(-1);
  }
  newnode->next = NULL;
  newnode->val = x;
  return newnode;
}
void SLPrint(SLNode* pphead)
{
  assert(pphead);
  SLNode* cur = pphead;
  while (cur)
  {
    printf("%d ", cur->val);
    cur = cur->next;
  }
  printf("NULL");
  printf("\n");
}
void SLPushFront(SLNode** pphead, SLDateType x)
{
  SLNode* newnode = BuySLnewnode(x);
  if (*pphead == NULL)
  {
    *pphead = newnode;
  }
  else
  {
    newnode->next = *pphead;
    (*pphead) = newnode;
  }
}
void SLPushBack(SLNode** pphead, SLDateType x)
{
  SLNode* newnode = BuySLnewnode(x);
  if (*pphead == NULL)
  {
    *pphead = newnode;
    return;
  }
  SLNode* tail = *pphead;
  while (tail->next)
  {
    tail = tail->next;
  }
  tail->next = newnode;
}
void SLPopFront(SLNode** pphead)
{
  assert(*pphead);
  if((*pphead)->next==NULL)
  {
    free(*pphead);
    *pphead = NULL;
    return;
  }
  SLNode* tmp = (*pphead)->next;
  free(*pphead);
  *pphead = tmp;
}
void SLPopBack(SLNode** pphead)
{
  assert(*pphead);
  if ((*pphead)->next == NULL)
  {
    free(*pphead);
    *pphead = NULL;
    return;
  }
  SLNode* Prevtail = *pphead;
  SLNode* tail = *pphead;
  while (tail->next)
  {
    Prevtail = tail;
    tail = tail->next;
  }
  free(tail);
  Prevtail->next = NULL;
}
SLNode* SLFind(SLNode* pphead, SLDateType x)
{
  assert(pphead);
  SLNode* cur = pphead;
  while (cur)
  {
    if (cur->val == x)
    {
      return cur;
    }
    cur = cur->next;
  }
  return NULL;
}
//在pos之前插入
void SLInsert(SLNode** pphead, SLNode* pos, SLDateType x)
{
  assert(*pphead);
  assert(pos);
  if (pos == *pphead)
  {
    SLPushFront(pphead, x);
    return;
  }
  SLNode* newnode = BuySLnewnode(x);
  
  SLNode* cur = *pphead;
  while (cur->next)
  {
    if (cur->next == pos)
    {
      newnode->next = cur->next;
      cur->next = newnode;
      return;
    }
    cur = cur->next;
  }
}
//在pos之后插入
void SLInsertBack(SLNode* pos, SLDateType x)
{
  assert(pos);
  SLNode * newnode = BuySLnewnode(x);
  newnode->next = pos->next;
  pos->next = newnode;
}
//删除pos之后的节点
void SLEraseBack(SLNode* pos)
{
  assert(pos);
  assert(pos->next);
  if (pos->next->next == NULL)
  {
    pos->next = NULL;
    return;
  }
  SLNode* cur = pos->next->next;
  free(pos->next);
  pos->next = cur;
  cur = NULL;
}
//删除pos之前的节点
void SLErase(SLNode** pphead, SLNode* pos)
{
  assert(pos);
  assert(pos != *pphead);
  if (pos== (*pphead)->next)
  {
    free(*pphead);
    (*pphead) = pos;
    return;
  }
  SLNode* cur = *pphead;
  while (cur)
  {
    if (cur->next->next == pos)
    {
      free(cur->next);
      cur->next = pos;
      return;
    }
    cur = cur->next;
  }
}
三.性能分析

与顺序表相比:

优点:

1.按需申请,没有空间浪费;

2.头插头删效率高

缺点:

1.不支持下标随机访问

2.尾插尾删效率低

目录
相关文章
|
2月前
|
算法 数据处理 C语言
C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合
本文深入解析了C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合,旨在帮助读者掌握这一高效的数据处理方法。
58 1
|
2月前
|
存储 算法 搜索推荐
【趣学C语言和数据结构100例】91-95
本文涵盖多个经典算法问题的C语言实现,包括堆排序、归并排序、从长整型变量中提取偶数位数、工人信息排序及无向图是否为树的判断。通过这些问题,读者可以深入了解排序算法、数据处理方法和图论基础知识,提升编程能力和算法理解。
57 4
|
2月前
|
存储 机器学习/深度学习 搜索推荐
【趣学C语言和数据结构100例】86-90
本文介绍并用C语言实现了五种经典排序算法:直接插入排序、折半插入排序、冒泡排序、快速排序和简单选择排序。每种算法都有其特点和适用场景,如直接插入排序适合小规模或基本有序的数据,快速排序则适用于大规模数据集,具有较高的效率。通过学习这些算法,读者可以加深对数据结构和算法设计的理解,提升解决实际问题的能力。
51 4
|
2月前
|
存储 算法 数据处理
【趣学C语言和数据结构100例】81-85
本文介绍了五个经典算法问题及其C语言实现,涵盖图论与树结构的基础知识。包括使用BFS求解单源最短路径、统计有向图中入度或出度为0的点数、统计无向无权图各顶点的度、折半查找及二叉排序树的查找。这些算法不仅理论意义重大,且在实际应用中极为广泛,有助于提升编程能力和数据结构理解。
54 4
|
2月前
|
算法 数据可视化 数据建模
【趣学C语言和数据结构100例】76-80
本文介绍了五种图论算法的C语言实现,涵盖二叉树的层次遍历及广度优先搜索(BFS)和深度优先搜索(DFS)的邻接表与邻接矩阵实现。层次遍历使用队列按层访问二叉树节点;BFS利用队列从源节点逐层遍历图节点,适用于最短路径等问题;DFS通过递归或栈深入图的分支,适合拓扑排序等场景。这些算法是数据结构和算法学习的基础,对提升编程能力和解决实际问题至关重要。
56 4
|
2月前
|
存储 算法 vr&ar
【趣学C语言和数据结构100例】71-75
本文介绍了五个C语言数据结构问题及其实现,涵盖链表与二叉树操作,包括按奇偶分解链表、交换二叉树左右子树、查找节点的双亲节点、计算二叉树深度及求最大关键值。通过递归和遍历等方法,解决了理论与实际应用中的常见问题,有助于提升编程能力和数据结构理解。
47 4
|
2月前
|
存储 算法 C语言
【趣学C语言和数据结构100例】66-70
本书《趣学C语言和数据结构100例》精选了5个典型的数据结构问题及C语言实现,涵盖链表与数组操作,如有序集合的集合运算、有序序列表的合并、数组中两顺序表位置互换、三递增序列公共元素查找及奇偶数重排。通过详细解析与代码示例,帮助读者深入理解数据结构与算法设计的核心思想,提升编程技能。
39 4
|
14天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
22小时前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
19 5
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
76 5

热门文章

最新文章

下一篇
开通oss服务