C语言实现对顺序表和链表的增删改【数据结构/初阶】

简介: C语言实现对顺序表和链表的增删改【数据结构/初阶】

1. 线性表

1.1 概念

线性表(linear list)是若干个具有相同特性的数据元素的有限序列,其本质是数组。

1.2 对线性的理解

这里的线性指的是逻辑上的连续,而不是物理空间上的连续。也就是说,以后所有关于链表的图示都是想象出来的,实际上在内存中不是这样的。当然顺序表在内存中的形式和图示相同。

2. 顺序表

2.1 概念

顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改。

顺序表分为两大类:

  1. 静态顺序表:使用定长数组存储。
  2. 动态顺序表:使用动态开辟的数组存储。

两者孰优孰劣一眼便知

顺序表的静态存储

#define N 100
typedef int SeqDataType;
//*typedef定义数据类型,方便以后修改数据类型
typedef struct SeqList
{
  SLDataType array[N]; // 定长数组
  int size; // 有效数据的个数
}SeqList;

顺序表的静态存储

typedef int SeqDataType;
typedef struct SeqList
{
  SLDataType* array; // 指向动态开辟的内存
  int size; // 有效数据个数
  int capacity; // 容量空间的大小(新增)
}SeqList;

在增加数据之前,需要通过比较容量计数器(capacity)和数据个数(size),判断是否要增加开辟的内存

2.2 实例

由于顺序表实际上就是数组,所以对顺序表的增删查改就是对数组的增删查改。下面给出代码。

2.2.0 命名习惯

在使用顺序表/链表等数据结构时,常常有以下命名习惯(只取一种)

以最常使用的驼峰命名法为例

中文名 接口
顺序表 SeqList
初始化 Init
头部 Front
尾部 Back
Push
Pop
Find
Modify
插入 Insert
检查 Check

2.2.1 SeqList.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
typedef int SeqDataType;
typedef struct SeqList
{
  SeqDataType* data;
  int size;
  int capacity;
}SeqList;
void SeqListInit(SeqList* sq);//初始化
void SeqListDestory(SeqList* sq);//销毁内存
void SeqListPrint(SeqList* sq);//打印
void SeqCheckCapacity(SeqList* sq);//检查容量
void SeqListPushBack(SeqList* sq, SeqDataType x);//尾增
void SeqListPushFront(SeqList* sq, SeqDataType x); //头增
void SeqListPopBack(SeqList* sq, SeqDataType x);//尾删
void SeqListPopFront(SeqList* sq, SeqDataType x);//头删
void SeqListInsert(SeqList* sq, SeqDataType x);//任意位置插入
void SeqListErase(SeqList* sq, SeqDataType x);//任意位置删除

2.2.2 SeqList.h(核心功能实现)

#define _CRT_SECURE_NO_WARNINGS 1
#include"SeqList.h"
//初始化
void SeqListInit(SeqList* sq)
{
  assert(sq);
  sq->data = NULL;
  sq->capacity = 0;
  sq->size = 0;
}
//销毁内存
void SeqListDestory(SeqList* sq)
{
  assert(sq);
  free(sq->data);
  sq->data = NULL;
  sq->capacity = 0;
}
//打印
void SeqListPrint(SeqList* sq)
{
  assert(sq);
  for (int i = 0; i < sq->size; i++)
  {
    printf("%d ", sq->data[i]);
  }
  printf("\n");
}
//检查容量
void SeqCheckCapacity(SeqList* sq)
{
  assert(sq);
  if (sq->capacity == sq->size)//增容
  {
    int Newcapacity = sq->capacity == 0 ? 4 : sq->capacity * 2;//如果容量为0,则开辟4个
    SeqDataType* NewArr = realloc(sq->data, sizeof(SeqDataType) * Newcapacity);
    if (NULL == NewArr)
    {
      perror("realloc\n");
      return;
    }
    sq->data = NewArr;//更新内存地址
    sq->capacity = Newcapacity;//更新容量
  }
}
//尾增
void SeqListPushBack(SeqList* sq, SeqDataType x)
{
  assert(sq);
  SeqCheckCapacity(sq);
  sq->data[sq->size] = x;
  sq->size++;
}
//头增
void SeqListPushFront(SeqList* sq, SeqDataType x)
{
  assert(sq);
  SeqCheckCapacity(sq);
  int end = sq->size ;
  while (end>= 0)
  {
    sq->data[end] = sq->data[end - 1];//数据往后移动一位
    end--;
  }
  sq->data[0] = x;
  ++sq->size;
}
//尾删
void SeqListPopBack(SeqList* sq, SeqDataType x)
{
  assert(sq);
  --sq->size;
}
//头删
void SeqListPopFront(SeqList* sq, SeqDataType x)
{
  assert(sq);
  int start = 0;
  while (start++ <= sq->size - 1)
  {
    sq->data[start] = sq->data[start + 1];//数据向前移动一位
  }
  sq->size--;
}
//查找
void SeqListFind(SeqList* sq, SeqDataType x)
{
  assert(sq);
  for (int i = 0; i < sq->size; i++)
  {
    if (sq->data[i] == x)
    {
      printf("找到了\n");
      break;
    }
  }
}
//任意位置插入
void SeqListInsert(SeqList* sq, SeqDataType x, int adr)
{
  assert(sq);
  SeqCheckCapacity(sq);
  int end = sq->size;
  while (end >= adr)
  {
    sq->data[end] = sq->data[end - 1];//移动数据
    end--;
  }
  sq->data[adr] = x;//插入数据
  sq->size++;
}
//任意位置删除
void SeqListErase(SeqList* sq, int adr)
{
  assert(sq);
  int begin = adr;
  while (begin + 1 <= sq->size - 1)
  {
    sq->data[begin] = sq->data[begin + 1];
    begin++;
  }
  sq->size--;
}

2.2.3 test.c(测试用例)

#define _CRT_SECURE_NO_WARNINGS 1
#include"SeqList.h"
void test1()
{
  SeqList s;
  SeqListInit(&s);
  SeqListPushBack(&s, 1);
  SeqListPushBack(&s, 2);
  SeqListPushBack(&s, 3);
  SeqListPushBack(&s, 4);
  SeqListPushFront(&s, 4);
  SeqListPushFront(&s, 4);
  SeqListPushFront(&s, 4);
  //先放一组数据进去
  SeqListPrint(&s);
  //打印
  SeqListInsert(&s, 0, 2);
  SeqListPrint(&s);
  //在第三个位置插入0
  SeqListErase(&s, 5);
  //删除第五个位置的数据
  SeqListPrint(&s);
  SeqListDestory(&s);
}
int main()
{
  test1();
  return 0;
}

2.4 顺序表的特点

  1. 增容有两种情况:一是在原内存后增容;二是空间不足,需要重新开辟内存,将原内存中的数据拷贝一份过去。这样不断拷贝数据,释放内存,会消耗机器性能。
  2. 头部或在中间插入或删除数据,时间复杂度为O(N)
  3. 增容一般以1.5倍或2倍增长,会造成一定程度上的空间浪费

链表能解决上述问题

3. 单向不带头非循环链表

3.1 概念

逻辑结构由链表中的指针连接而成,物理内存中并不是连续的

链表分为:单双向、带/不带头、循环/非循环。由这些不同的结构组合一共有8种不同的链表结构。

此处仅介绍最简单(无头单向非循环链表)和最复杂(带头双向循环链表)的两种链表,前者是考试、OJ题的模板,后者是实际应用中最常使用的链表。其他类型的链表也会在学习它们的过程中掌握。

3.2 链表的实现

说明:为了篇幅,此处仅放实现功能的核心代码

定义链表

typedef int SListDataType;
typedef struct SListNode
{
  SListDataType data;
  struct SListNode* next;//指针变量,指向下一个节点的地址
}SListNode;
SListNode* s = NULL;//注意这里是一个指针变量
//这么做的目的是连接每个节点,因为next也是一个指针变量

3.2.1 增加节点

//新增新节点
SListNode* CrateSListNode(SListDataType x)
{
  SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
  newnode->data = x;
  newnode->next = NULL;
  return newnode;//返回开辟的地址
}

3.2.2 头插

//头插
void SListNodePushFront(SListNode** pplist, SListDataType x)
{
  SListNode* newnode = CrateSListNode(x);
  newnode->next = *pplist;
  *pplist = newnode;
}

将新增的空间的next指向原本的第一个节点的地址。因为是用二级指针接收的节点的地址,所以pplist就是节点本身的地址,由于这个函数不返回新增节点的地址newnode ,则还需将newnode给pplist 。相当于原来的地址更新了。

3.2.3 尾插

//尾插
void SListNodePushBack(SListNode** pplist, SListDataType x)
{
  SListNode* newnode = CrateSListNode(x);
  //第一个节点为空
  if (*pplist == NULL)
  {
    *pplist = newnode;
  }
  else
  {
    SListNode* tail = *pplist;//找尾
    while (tail->next != NULL)
    {
      tail = tail->next;
    }
    tail->next = newnode;//新尾
  }
}

这里最需要注意的是,链表有一个缺点:不能像顺序表(数组)一样随机访问,因为链表的内存不是连续的,所以要找到某一个节点,每次都要从头开始,根据条件让迭代停止,保存尾节点tail。(迭代即循环遍历)

所以要在链表尾端插入数据,需要先找尾节点,然后将尾节点的next指针变量指向新增节点的地址。(通过创建节点函数新增的节点的next指针已经是NULL了)

要注意判断第一个节点是否为空的条件

3.2.4 尾删

//尾删
void SListNodePopBack(SListNode** pplist)
{
  if (*pplist == NULL)//空链表
  {
    return;
  }
  else if ((*pplist)->next == NULL)//只有一个节点
  {
    free(*pplist);
    *pplist = NULL;
  }
  else
  {
    SListNode* pre = NULL;
    SListNode* tail = *pplist;
    while (tail != NULL)//找尾
    {
      pre = tail;//先保存前一个节点的地址再迭代
      tail = tail->next;
    }
    free(tail);
    pre->next = NULL;
  }
}

尾删的主要步骤就是 1. 找尾(要删的节点) 2. 将尾节点的前一个节点pre的next指针变量置空。要更新pre的值,就要在迭代循环中,在tail更新之前将它保存,即为pre的值。

3.2.5 在pos后插入节点

//在pos后插入节点
void SListNodeInsertAfter(SListNode** pplist, SListNode* pos, SListDataType x)
{
  assert(pos);
  SListNode* newnode = CrateSListNode(x);
  newnode->next = pos->next;
  pos->next = newnode;
}

此处代码的实现最重要的是不能丢掉pos下一个节点的位置,即pos->next

一般有两种办法:1. 另外用一个变量(通常命名为next)保存 2. 在它改变之前将其值赋给需要的变量。两种办法因情况使用。

3.2.6 在pos前插入节点(仅示例)

//在pos前插入节点
void SListNodeInsertBefore(SListNode** pplist, SListNode* pos, SListDataType x)
{
  assert(pos);
  SListNode* newnode = CrateSListNode(x);
  if (*pplist == pos)
  {
    SListNodePushFront(pplist, x);
  }
  else
  {
    SListNode* prev = NULL;
    SListNode* cur = *pplist;
    while (cur != pos)//找尾
    {
      prev = cur;//同等地位,prev不要往下访问成员 
      cur = cur->next;
    }
    newnode->next = pos;
    prev->next = newnode;
  }
}

在pos前插入节点需要判断节点是否为首节点,还要找尾,因为是在pos前插入,所以要找到pos前的节点prev。其他操作和3.2.5一样。

在pos前插入节点,在实际中是不会使用的,因为它的效率没有在pos后插入节点高。

3.2.7 删除pos后的节点

//删除pos后的节点
void SListNodeEraseAfter(SListNode* pos)
{
  assert(pos);
  if (pos->next == NULL)
  {
    return;
  }
  else
  {
    SListNode* next = pos->next;
    pos->next = next->next;
    free(next);
  }
}

删除pos后的节点,思路是:保存pos的下一个节点(next)的地址,这个节点也是要被删除的。将next的next指针变量的值给pos位置的next指针,这样pos后面的节点与两部分都没有联系了,最后free掉它。

相同思路的另一种解释:从pos出发:pos->next->next,访问到pos后第二个节点的地址,相当于走了两步,直接跨过了中间要删除的节点,最后将它free掉即可。

4. 双向带头循环链表

双向带头循环链表虽然结构最复杂,但结构的复杂度使得在处理它时相比于最简单的结构(单向不带头非循环链表)更为简单。

结构图示:

其中哨兵位头结点不存放有效数据。

如果不看这个头结点,单看循环结构的话,链表是不存在头或尾的,不论在何处插入或删除数据,都相当于在链表中间插入,为了方便处理,人们把头结点视为链表的起点。

4.1 核心代码

4.1.1 查找函数

查找函数通过输入的链表起始地址和位置,找到该位置则返回地址。插入和删除函数都需要调用它。

struct ListNode* ListNodeFind(struct ListNode* plist, int pos)
{
  if (pos < 0)
    return -1;
  assert(plist);
  int count = 0;
  struct ListNode* cur = plist->next;
  while (cur != plist && count < pos)
  {
    count++;
    cur = cur->next;
  }
  return cur;
}

4.1.2 初始化函数

在主函数测试时,结构体中什么都没有,所以需要让头结点的prev和next都指向自己。

//初始化链表,新增一个哨兵位节点
struct ListNode* ListNodeInit()
{
  struct ListNode* head = (struct ListNode*)malloc(sizeof(struct ListNode));
  //只有一个哨兵位节点,让它两个指针指向自己
  head->next = head;
  head->prev = head;
  return head;
}

4.1.3 创建新节点函数

在插入函数中需要被调用,返回开辟的地址

//创建新节点
struct ListNode* CreatListNode(LTDataType x)
{
  struct ListNode* newnode = (struct ListNode*)malloc(sizeof(struct ListNode));
  newnode->val = x;
  return newnode;
}

4.1.4 插入新节点函数

在任意位置插入新节点,需要调用2.3函数

4.1.5 删除节点函数

//删除任意位置的节点
void ListNodeErase(struct ListNode* pos, int Pos)
{
  assert(pos);
  pos = ListNodeFind(pos, Pos);
  struct ListNode* prev = pos->prev;
  struct ListNode* next = pos->next;
  prev->next = next;
  next->prev = prev;
  free(pos);
}

4.1.6 打印函数

void ListNodePrint(struct ListNode* plist)
{
  assert(plist);
  struct ListNode* cur = plist->next;
  while (cur != plist)//有效节点不等于哨兵位节点
  {
    printf("%d ", cur->val);
    cur = cur->next;
  }
  printf("\n");
}
//在任意位置插入新节点
void ListNodeInsert(LTDataType x, struct ListNode* pos, int Pos)
{
  assert(pos);
  pos = ListNodeFind(pos, Pos);
  struct ListNode* newnode = CreatListNode(x);
  struct ListNode* prev = pos->prev;
  //链接
  newnode->next = pos;
  pos->prev = newnode;
  prev->next = newnode;
  newnode->prev = prev;
}

5. 顺序表和链表的优缺点

5.1 顺序表

顺序表的优点:

  1. 能通过下标随机访问
  2. cpu高速缓存命中率更高

顺序表的缺点:

  1. 不断增容,会造成性能消耗,也可能存在一定的空间浪费。
  2. 头部或中间插入数据需要挪动数据,时间复杂度为O(N)

5.2 链表

链表的优点:

  1. 按需申请开辟内存,不存在空间浪费
  2. 插入数据不需要挪动数据,时间复杂度为O(1)

链表的缺点

  1. 不能通过下标随机访问

更新日志

4/12/2022
4/19/2022
新增双向带头循环链表
顺序表和链表的优缺点对比
Man9o
目录
相关文章
|
2月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
262 9
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
67 4
|
2天前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
20 5
|
2天前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】顺序表的基本运算(头歌实践教学平台习题)【合集】
本文档介绍了线性表的基本运算任务,涵盖顺序表和链表的初始化、销毁、判定是否为空、求长度、输出、查找元素、插入和删除元素等内容。通过C++代码示例详细展示了每一步骤的具体实现方法,并提供了测试说明和通关代码。 主要内容包括: - **任务描述**:实现顺序表的基本运算。 - **相关知识**:介绍线性表的基本概念及操作,如初始化、销毁、判定是否为空表等。 - **具体操作**:详述顺序表和链表的初始化、求长度、输出、查找、插入和删除元素的方法,并附有代码示例。 - **测试说明**:提供测试输入和预期输出,确保代码正确性。 - **通关代码**:给出完整的C++代码实现,帮助完成任务。 文档
20 5
|
16天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
1月前
|
存储 算法 C语言
【C语言】深入浅出:C语言链表的全面解析
链表是一种重要的基础数据结构,适用于频繁的插入和删除操作。通过本篇详细讲解了单链表、双向链表和循环链表的概念和实现,以及各类常用操作的示例代码。掌握链表的使用对于理解更复杂的数据结构和算法具有重要意义。
556 6
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
78 5
|
2月前
|
存储
顺序表和链表(2)
【10月更文挑战第23天】
顺序表和链表(2)
|
2月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
63 4
|
2月前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
112 16