【数据结构】顺序表和链表2(上)

简介: 【数据结构】顺序表和链表2(上)

 对于上一篇文中说到的顺序表,我们不难发现,它本身有很多限制


1. 中间/头部的插入删除,时间复杂度为O(N)

2. 增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。

3. 增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们再继续插入了5个数据,后面没有数据插入了,那么就浪费了95个数据空间


为了解决这些问题,我们给出了新的数据结构——链表。


二、线性表的链式结构——链表


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


1.链表的分类

实际中链表的结构非常多样,以下情况组合起来就有8种链表结构:

1.单向或者双向

e6989531690647de8661fbf1536a95e0.png

2.带头或者不带头

23eae7ea4f0a4622b7587c231468e150.png

3.循环或者非循环

88df1d22cb0f4fe1bacc88e78096c87e.png

虽然有这么多种链表,但是最常用的只有两种:

6bb5f1ba549e45a9b3bc34f552d861d8.png

无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。


2. 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单。


2.链表的实现


(1)单链表的接口实现

1.存储结构

typedef int SLTDateType;
typedef struct SListNode
{
  SLTDateType data;//数据域
  struct SListNode* next;//指针域
}SListNode;


2.初始化与销毁

由于链表的结构是由连接起来的节点组成的,所以我们并不需要对其进行初始化。

链表的每一个节点都是单独malloc出来的,因此我们要销毁这些节点,只能遍历链表,依次释放。将链表释放完毕以后,最好将指向链表头节点的指针置空,防止野指针。

void SListDestroy(SListNode** pplist)
{
  assert(pplist);
  SListNode* cur = *pplist;
  while (cur)//遍历并销毁
  {
    SListNode* next = cur->next;
    free(cur);
    cur = next;
  }
  *pplist = NULL;
}


3.插入数据

插入数据也可以分为头插、尾插、在任意位置前插入,在任意位置后插入。但是不管是在哪里插入,都是需要动态申请一个节点的,然后根据插入位置不同链接不同,所以我们把申请节点封装成一个函数

SListNode* BuySListNode(SLTDateType x)
{
  SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
  if (newnode == NULL)
  {
    perror("malloc:");
    exit(-1);
  }
  newnode->data = x;
  newnode->next = NULL;
  return newnode;
}


将需要的数据放入节点内部,然后执行插入操作的时候只需要将新节点链接上链表即可


  • 头插

头插就是让新节点做头,新节点的next指向链表原来的头。

void SListPushFront(SListNode** pplist, SLTDateType x)
{
  SListNode* newnode = BuySListNode(x);
  newnode->next = *pplist;
  *pplist = newnode;
}


  • 尾插

在单链表的最后一个节点后面插入一个新节点,我们需要遍历找到单链表的尾结点,然后将新节点链接上去

void SListPushBack(SListNode** pplist, SLTDateType x)
{
  assert(pplist);
  SListNode* newnode = BuySListNode(x);
  if (*pplist == NULL)//判断是否为第一个节点
  {
    *pplist = newnode;
  }
  else
  {
    SListNode* tail = *pplist;
    while (tail->next != NULL)//找尾结点
    {
      tail = tail->next;
    }
    tail->next = newnode;
  }
}


  • 在任意位置之前插入

要在pos位置之前插入数据,我们需要找到pos位置之前的节点指针prev,然后在prev的位置之后插入新节点,原理与尾插相同。但是当传入的pos位置为头节点的时候,需要把情况单独拿出来说,传入头节点本质上就是头插,因此直接调动头插函数即可。

void SListInsertBefore(SListNode** pplist, SListNode* pos, SLTDateType x)
{
  assert(pplist);
  assert(pos);
  if (pos == *pplist)
  {
    SListPushFront(pplist, x);
  }
  else
  {
    SListNode* prev = *pplist;
    while (prev->next != pos)//找pos的前一个节点
    {
      prev = prev->next;
      assert(prev);//暴力检查pos传错了的情况
    }
    SListNode* newnode = BuySListNode(x);
    prev->next = newnode;
    newnode->next = pos;
  }
}


  • 在任意位置之后插入

由于已经传入一个位置pos,并且在pos后面链接,所以原理与尾插相同。

void SListInsertAfter(SListNode* pos, SLTDateType x)
{
  assert(pos);
  SListNode* newnode = BuySListNode(x);
  newnode->next = pos->next;
  pos->next = newnode;
}


4.删除数据


  • 头删

头删数据需要改变头节点,所以定义一个del指针存放需要删除的节点,然后改变头节,最后释放del指向的节点。

void SListPopFront(SListNode** pplist)
{
  assert(pplist);
  assert(*pplist);
  SListNode* del = *pplist;
  *pplist = (*pplist)->next;
  free(del);
  del = NULL;
}


  • 尾删

对于一个正常的单链表,删除尾节点,首先需要找到尾,找到尾节点并记录前一个结点,将尾节点释放,然后将尾节点的前一个节点置空,但是对于只有一个元素的节点的情况,需要单独考虑,将该节点的指针指向的next置空,然后释放该节点即可。

void SListPopBack(SListNode** pplist)
{
  assert(pplist);
  if ((*pplist)->next == NULL)
  {
    free(*pplist);
    *pplist = NULL;
  }
  else
  {
    SListNode* tail = *pplist;
    while (tail->next->next != NULL)//找到尾结点的前一个结点
    {
      tail = tail->next;
    }
    free(tail->next);
    tail->next = NULL;
  }
}


  • 在任意位置删除

首先判断要删除的是否为头节点,如果是,直接调用头删函数,如果不是,那么就遍历链表,找到pos指针的前一个节点prev,然后将prev和pos的下一个节点链接起来,然后删除prev即可。

void SListEraseAfter(SListNode** pplist, SListNode* pos)
{
  assert(pos);
  if (pos == *pplist)
  {
    SListPopFront(pplist);
  }
  else
  {
    SListNode* prev = *pplist;
    while (prev->next != pos)
    {
      prev = prev->next;
    }
    SListNode* tmp = prev->next;
    prev->next = prev->next->next;
    free(tmp);
    tmp = NULL;
  }
}


相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
65 4
|
10天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
70 5
|
2月前
|
存储
顺序表和链表(2)
【10月更文挑战第23天】
顺序表和链表(2)
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
116 4
|
2月前
|
存储 算法 数据管理
顺序表和链表(1)
【10月更文挑战第22天】
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
53 0
|
2月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
79 0
|
7月前
|
存储 SQL 算法
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表

热门文章

最新文章