【数据结构和算法】认识线性表中的链表,并实现单向链表(下)

简介: 【数据结构和算法】认识线性表中的链表,并实现单向链表(下)

6.在pos结点位置删除数据(删除pos结点)

如图所示:

代码如下:

//在pos位子删除数据
void ListErase(List** ps, List* pos) {
  assert(ps);
  assert(pos);
  if (*ps == pos) {
    ListPopFront(ps);
  }
  else {
    List* next = *ps;
    while (next->next != pos) {
      next = next->next;
    }
    //这个时候next->next == pos
    next->next = next->next->next;
    /*free(next->next);*/
    free(pos);
    pos = NULL;
  }
}

7.删除pos位置之后一个结点

如图所示:

代码如下:

//在pos位置之后一位删除数据
void ListEraseAfter(List* pos) {
  assert(pos);
  List* next = pos->next;//将pos 的下一个结点赋值给next
  if (next != NULL) {
    pos->next = pos->next->next;//表示pos的下一个的下一个结点的地址赋值给pos的指针域  实质上是将pos的下一个结点给跳过
    free(next);  //将pos的下一个结点给free释放
    next = NULL;  //next指向为NULL  防止野指针
  }
}

8.摧毁链表

代码如下:

//链表的摧毁  直接将头指针指针域指向NULL
void ListDestory(List** ps) {
  //assert(ps);  //防止空链表
  一个结点一个结点释放
  //List* next = *ps;
  //while (next) {
  //  List* cur = next->next;
  //  free(next);
  //  next = cur;
  //}
  *ps = NULL;
}

因为是二级指针,所以直接 *ps=NULL 即可,或者一个一个free

三、完整代码

1.LinkList.h

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
#include<malloc.h>
#include<assert.h>
typedef int SLDataType;
//单向链表的实现、
typedef struct ListNode {
  SLDataType data;//数据域
  struct ListNode* next;//指针域
}List;
//打印单链表
void ListPrint(List* ps);
//单链表的尾插
void ListPushBack(List** ps, SLDataType data);
//单链表的头插
void ListPushFront(List** ps, SLDataType data);
//单链表的尾删
void ListPopBack(List** ps);
//单链表的头删
void ListPopFront(List** ps);
//单链表的查找
List* ListFind(List* ps);
//在pos位置上插入数据
void ListInsertBefore(List** ps, SLDataType x, List* pos);
//在pos位置之后插入数据
void ListInsertAfter(List** ps, SLDataType x, List* pos);
//在pos位子删除数据
void ListErase(List** ps, List* pos);
//在pos位置之后一位删除数据
void ListEraseAfter(List* pos);
//单链表的摧毁
void ListDestory(List** ps);

2.LinkList.c

#define _CRT_SECURE_NO_WARNINGS
#include"单向链表.h"
//链表的使用,在插入上面
// 如果是尾部插入,如果是空链表直接将新节点给ps 是先找到链表尾部,然后创建新节点,连接即可
// 如果是头部插入,先进行断言判空,之后创建新节点,将新节点的数据
// new->next=ps  这个是找到对应的位置,连接起来
// ps=new;  将新节点的信息传递给ps,这样ps还是头节点
// 
// 
// 
// 
// 
//进行单链表的实现
//初始化链表
void InitList(List* ps) {
  ps->data = 0;
  ps->next = NULL;
}
//打印单链表
void ListPrint(List* ps) {
  List* cur = ps;
  while ((cur) != NULL) {
    printf("%d -> ", cur->data);
    cur = cur->next;
  }
  printf("NULL\n");
}
//创建一个新节点
List* CreateNode(SLDataType x) {
  List* newNode = (List*)malloc(sizeof(List));
  if (newNode == NULL) {
    perror("malloc fail\n");
    exit(-1);
  }
  else {
    newNode->data = x;
    newNode->next = NULL;
  }
  return newNode;
}
//单链表的尾插
void ListPushBack(List** ps, SLDataType data) {
  //创建新的节点
  assert(ps);//断言
  List* newNode = CreateNode(data);
  if (*ps == NULL) {
    //说明是空链表
    *ps = newNode;
  }
  else {
    List* tail = *ps;
    while (tail->next != NULL) {
      tail = tail->next;
    }
    tail->next = newNode;
  }
}
//单链表的头插
void ListPushFront(List** ps, SLDataType data) {
  //先断言是否为空
  assert(ps);
  //将新地址指向头结点下一个next结点的地址,然后在用头结点指向新节点
  List* newNode = CreateNode(data);
  newNode->next = (*ps);  //new指向ps当前的位置,然后new是第一个位置了,将new赋值给ps,这样new就作为头部连接链表了
  (*ps) = newNode;//原本ps位置的数值不变,这样的话就成 new->next=ps,new数值在前,ps的数值在后
}
//单链表的尾删
void ListPopBack(List** ps) {
  assert(ps);//断言
  //三种情况
  //1.空链表
  //2.一个节点
  //3.多个节点
  if (*ps == NULL) {
    return;
  }
  //只有一个节点的情况为
  else if ((*ps)->next == NULL) {
    free(*ps); //如果只有一个头节点的话
    *ps = NULL;
  }
  else {
    //多个节点的情况下、
    List* tail = *ps;
    while (tail->next->next!= NULL) {
      tail = tail->next;
    }
    free(tail->next);
    tail->next= NULL;
  }
}
//单链表的头删
void ListPopFront(List** ps) {
  assert(ps);
  //1.空
  //2.非空
  if (*ps == NULL) {
    //为空
    return;
  }
  else {
    List* tail = (*ps)->next;//创建临时变量tail,将头节点之后的地址给tail
    free(*ps);//滞空头节点
    *ps = NULL;//可有可不有,接下来也要用
    *ps = tail;//将tail也就是ps的下一个List节点给ps
  }
}
//单链表的查找
List* ListFind(List* ps,SLDataType data) {
  //进行查找就是进行判断是否为空链表,为空直接返回
  if (ps == NULL) {
    printf("链表为空、无法查找\n");
    return;
  }
  List* tail = ps;
  while (tail != NULL) {//从头节点开始,进行循环,
    if (tail->data == data) {
      return tail;
    }
    tail = tail->next;
  }
  return tail;
}
//在pos位置上插入数据
void ListInsertBefore(List** ps, SLDataType x, List* pos) {
  //先判断是否为空
  assert(ps);
  assert(pos);
  //空链表排除
  //1.pos是第一个节点
  //2.pos不是第一个节点
  if (*ps == pos) {
    //是第一个节点,那就直接头插
    ListPushFront(ps, x);
  }
  else {
    List* prev = *ps;
    while (prev->next != pos) {
      prev = prev->next;
    }
    List* newnode = CreateNode(x);
    prev->next = newnode;
    newnode->next = pos;
  }
}
//在pos位置之后插入数据
void ListInsertAfter(List** ps, SLDataType x, List* pos) {
  assert(ps);
  //assert(pos);//断言
  List* newnode = CreateNode(x);
  newnode->next = pos->next;
  pos->next = newnode;
}
//在pos位子删除数据
void ListErase(List** ps, List* pos) {
  assert(ps);
  assert(pos);
  if (*ps == pos) {
    ListPopFront(ps);
  }
  else {
    List* next = *ps;
    while (next->next != pos) {
      next = next->next;
    }
    //这个时候next->next == pos
    next->next = next->next->next;
    /*free(next->next);*/
    free(pos);
    pos = NULL;
  }
}
//在pos位置之后一位删除数据
void ListEraseAfter(List* pos) {
  assert(pos);
  List* next = pos->next;//将pos 的下一个结点赋值给next
  if (next != NULL) {
    pos->next = pos->next->next;//表示pos的下一个的下一个结点的地址赋值给pos的指针域  实质上是将pos的下一个结点给跳过
    free(next);  //将pos的下一个结点给free释放
    next = NULL;  //next指向为NULL  防止野指针
  }
}
//链表的摧毁  直接将头指针指针域指向NULL
void ListDestory(List** ps) {
  //assert(ps);  //防止空链表
  一个结点一个结点释放
  //List* next = *ps;
  //while (next) {
  //  List* cur = next->next;
  //  free(next);
  //  next = cur;
  //}
  *ps = NULL;
}

3.test.c

#define _CRT_SECURE_NO_WARNINGS
#include"单向链表.h"
void test()
{
  List* phead=NULL;//作为头节点
  //单链表的尾插
  ListPushBack(&phead, 1);
  ListPushBack(&phead, 2);
  ListPushBack(&phead, 3);
  ListPushBack(&phead, 4);
  ListPushBack(&phead, 5);
  ListPrint(phead);
  ListPushFront(&phead, 1);
  ListPrint(phead);
  ListPopBack(&phead);
  ListPrint(phead);
  ListPopFront(&phead);
  ListPrint(phead);
  ListErase(&phead, phead->next);
  ListInsertAfter(&phead, 10, phead->next);
  ListEraseAfter(phead->next);
  ListPrint(phead);
  ListDestory(&phead);
}
int main()
{
  test();
  return 0;
}

总结

本文主要讲解了链表的分类是什么,两种常用的类型,无头单向非循环链表、有头双向循环链表,我们实现了无头单向非循环链表,这是比较简单的一种链表的实现,我们使用的是二级指针传参,当然使用一级指针传参也可以,主要实现函数为头尾插入,头尾删除,pos指定结点位置前后添加或者删除元素。

接下来,下文我们将跟大家介绍一下最常用链表的另一种形式,带头双向循环链表。

相关文章
|
6月前
|
存储 前端开发 Java
线性数据结构详解
本文介绍了线性数据结构中的核心概念——节点,以及基于节点构建的链表、队列和栈等重要数据结构。节点是计算机科学中基本的构建单元,包含数据和指向其他节点的链接。通过添加约束或行为,可以构建出单向链表、双向链表、队列和栈等复杂结构。
166 1
|
8月前
|
存储 Python
Python 实现单向链表,和单向链表的反转
链表是一种数据结构,每个节点存储相邻节点的位置信息。单链表中的节点仅存储下一节点的位置。通过Python实现单链表,定义`ListNode`类并关联节点可创建链表。例如,创建A-&gt;B-&gt;C的链表后,可通过反转函数`reverse`将链表反转为CBA。代码展示了如何实现和操作单链表。
166 6
Python 实现单向链表,和单向链表的反转
|
存储 JavaScript 前端开发
JavaScript实现单向链表
JavaScript实现单向链表
255 109
|
9月前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
376 5
java数据结构,线性表链式存储(单链表)的实现
文章讲解了单链表的基本概念和Java实现,包括头指针、尾节点和节点结构。提供了实现代码,包括数据结构、接口定义和具体实现类。通过测试代码演示了单链表的基本操作,如添加、删除、更新和查找元素,并总结了操作的时间复杂度。
java数据结构,线性表链式存储(单链表)的实现
|
存储
【初阶数据结构】深入解析单链表:探索底层逻辑(无头单向非循环链表)(一)
【初阶数据结构】深入解析单链表:探索底层逻辑(无头单向非循环链表)
146 1
|
机器学习/深度学习 算法 Java
[算法与数据结构] 谈谈线性查找法~
该文章详细介绍了线性查找法的基本概念与实现方法,通过Java代码示例解释了如何在一个数组中查找特定元素,并分析了该算法的时间复杂度。
【数据结构OJ题】复制带随机指针的链表
力扣题目——复制带随机指针的链表
134 1
【数据结构OJ题】复制带随机指针的链表

热门文章

最新文章