【数据结构】单链表

devtools/2024/11/7 3:44:06/

单链表

文章目录

  • 单链表
    • 定义
      • 单链表的优缺点
      • 用代码定义单链表
      • 初始化单链表
        • 不带头结点的单链表
        • 带头结点的单链表
    • 单链表的插入
      • 按位序插入(带头结点)
      • 指定结点的后插操作
      • 指定结点的前插操作
    • 单链表的删除
      • 按位序删除(带头节点)
      • 删除指定结点
    • 单链表的查找
      • 按位查找
      • 按值查找
    • 求单链表的长度
    • 单链表的建立
      • 尾插法
      • 头插法

定义

单链表:用链式存储的方式实现线性表

链式存储:用一组任意的存储单元存储线性表的数据元素,不要求逻辑上相邻的元素在物理位置上也相邻

单链表的优缺点

  • 优点:不要求大片连续的空间,该容量方便
  • 缺点:不可随机存取,要耗费一定的空间存放指针(每个结点除了存放数据元素外,还要存储指向下一个结点的指针)

用代码定义单链表

struct LNode{				//定义单链表结点类型ElemType data;			//每个结点存放一个数据元素struct LNode *next;		//指针指向下一个结点
};
typedef struct LNode LNode;
typedef struct LNode *LinkList;

typedef对上面的代码进行简化

typedef <数据类型> <别名>

typedef struct LNode {		// 定义单链表结点类型ElemType data;          // 每个结点存放一个数据元素struct LNode *next;     // 指针指向下一个结点
} LNode, *LinkList;

要表示一个单链表时,只需声明一个头指针 L ,指向单链表的第一个结点

LNode * L;	//声明一个指向单链表第一个结点的指针,强调这是一个结点LinkList L;	//声明一个指向单链表第一个结点的指针,强调这是一个单链表

初始化单链表

InitList(&L)初始化表,构造一个空的线性表L,分配内存空间

不带头结点的单链表
typedef struct LNode {			// 定义单链表结点类型ElemType data;              // 每个结点存放一个数据元素struct LNode *next;         // 指针指向下一个结点
} LNode, *LinkList;bool InitList(LinkList &L) {L = NULL;                  //空表,暂时还没有任何结点,防止出现脏数据return true;
}void test(){LinkList L;		// 声明一个指向单链表的指针,这里的L是结构体指针// 注意此处并没有创建一个结点    InitList(L);	//初始化一个空表//......后续代码....
}
带头结点的单链表
typedef struct LNode {          // 定义单链表结点类型ElemType data;              // 每个结点存放一个数据元素struct LNode *next;         // 指针指向下一个结点
} LNode, *LinkList;// 初始化一个单链表(带头结点)
bool InitList(LinkList &L){L = (LNode*) malloc(sizeof(LNode));	//分配头结点if (L == NULL) {			//内存不足,分配失败return false;}L->next = NULL;				//头结点后暂时没有存放数据return true;
}void test(){LinkList L;		//声明一个指向单链表的指针,这里的L是结构体指针//注意此处并没有创建一个结点InitList(L);	// 初始化一个空表//......后续代码....
}

不带头结点,写代码更麻烦,需要对第一个数据结点和后续数据结点的处理,需要用不同的代码逻辑,对空表和非空表的处理需要用不同的代码逻辑
带头结点,一套代码完成,实现更方便

单链表的插入

ListInsert(&L,i,e)插入操作,在表L中的第 i 个位置上插入指定元素 e

按位序插入(带头结点)

思路

  1. 在表 L 中的第 i 个位置上插入指定元素 e,那么首先要找到第 i-1 个结点,将新节点插入其后,如果是插入到第 1 个位置,那么头节点可以视为第 0 个结点
  2. 新结点后继为所寻结点的后继,新结点成为所寻结点的后继
typedef struct LNode {ElemType data;struct LNode *next;
} LNode, *LinkList;
bool ListInsert(LinkList &L, int i, ElemType e)
{if (i < 1) {return false;}LNode *p;   // 结构体指针p, 指向当前扫描到的结点int j = 0;  // 当前p指向的是第j个结点p = L;      // L 指向头结点,头结点是第0个结点(不存数据)// 循环找到第 i - 1 个结点// 非空表示不能到链表尽头了,j < i - 1,表示要找到第 i - 1 个节点就停下来了while (p != NULL && j < i - 1) {      p = p->next;j++;}// 到这里 p 指向了第 i - 1 个节点// 校验发现第 i - 1个节点是空的,这就表示原链表只有 i - 2 个节点,第 i - 2 个节点的后继是空指针。if (p == NULL) {            return false;}LNode *s = (LNode *)malloc(sizeof(LNode));// malloc 返回值必须校验if (s == NULL) {return false;}s->data = e;s->next = p->next; p->next = s;    // 将结点s连到p之后//注意上面两句代码的执行顺序不能颠倒return true;    // 插入成功
}

时间复杂度分析

  • 最好情况:新元素插入到表头;最好时间复杂度 = O ( 1 ) O(1) O(1)
  • 最坏情况:新元素插入到表尾;最坏时间复杂度 = O ( n ) O(n) O(n)
  • 平均情况:新元素插入到任何一个位置的概率相同; 平均时间复杂度 = O ( n ) O(n) O(n)

指定结点的后插操作

// 在 p 结点后插入元素 e
bool InsertNextNode(LNode* p, ElemType e)
{if (p == NULL) {return false;}LNode *s = (LNode*)malloc(sizeof(LNode));// 判定内存是否申请成功if (s == NULL) {return false;}// 链表插入数据分三步,填数,建后继链,断前驱链s->data = e;s->next = p->next;p->next = s;return true;
}

时间复杂度 O ( 1 ) O(1) O(1)

指定结点的前插操作

方法1:常规算法

bool InsertPriorNode(Linklist L, LNode *p, ElemType e)
{if (p == NULL) {return false;}LNode * pre = L;   // 前驱结点// 遍历循环查找while (pre != NULL) {if (pre->next == p) {break;}pre = pre->next;}// 如果找不到,那就是空值if (pre == NULL) {return false;}// 直接调用指定结点后插元素,在p的前一个结点插入,即是在p的前驱结点的后一个插入return InsertNextNode(pre, e);
}

时间复杂度 O ( n ) O(n) O(n)

方法2:交换节点算法

思路:直接对指定节点进行后插操作,交换新节点与指定节点的数据域

bool InsertPriorNode(LNode *p, ElemType e)
{if (p == NULL) {return false;}LNode *s = (LNode*)malloc(sizeof(LNode));// 判定内存是否申请成功if (s == NULL) {return false;}// 解链,重新成链s->next = p->next;p->next = s;// 数据交换swaps->data = p->data;p->data = e;return true;
}

时间复杂度 O ( 1 ) O(1) O(1)

单链表的删除

ListDelete(&L,i,&e)删除操作,删除表L中第 i 个位置的元素,并用 e 返回删除元素的值

按位序删除(带头节点)

思路:头结点可以看作“第0个”结点,遍历找到第 i-1 个结点,将其指针指向第 i+1 个结点,并释放第 i 个结点

bool ListDelete(Linklist &L, int i, ElemType &e)
{if (i < 1) {return false;}LNode *p;   // 结构体指针p, 指向当前扫描到的结点int j = 0;  // 当前 p 指向的是第 j 个结点p = L;      // L 指向头结点,头结点是第0个结点(不存数据)// 循环找到第 i-1 个结点while (p != NULL && j < i - 1) {      p = p->next;j++;}if (p == NULL) {return false;   // i值不合法}       // 第i-1个结点后没有其他结点,不存在第i个结点if (p -> next == NULL) {return false;   }// 到这里的代码都和指定位序插入结点相同LNode *q = p->next;       // 令q指向被删除结点,即临时变量e = q->data;              // 用e把返回值带回来p->next = q->next;        // 将*q结点从链中断开free(q);                  // 释放结点的存储空间return true;              // 删除成功
}

时间复杂度 O ( n ) O(n) O(n)

删除指定结点

类似指定结点的插入操作的方法

单链表的查找

按位查找

GetElem(L,i):按位查找操作,获取表L中第 i 个位置的元素的值

//按位查找,返回第 i 个元素(带头结点)
LNode * GetElem(LinkList L, int i){if(i < 0){return NULL;}LNode *p;   // 结构体指针p, 指向当前扫描到的结点int j = 0;  // 当前 p 指向的是第 j 个结点p = L;      // L 指向头结点,头结点是第0个结点(不存数据)while(p != NULL && j < i){p = p->next;j++;}return p;
}

平均时间复杂度 O ( n ) O(n) O(n)

按值查找

LocateElem(L,e):按值查找操作,在表L中查找具有给定关键字值的元素

LNode * LocateElem(LinkList L,ElemType e){LNode *p = L->next;    //从第 1 个结点开始查找数据域为 e 的结点while(p != NULL && p->data != e){p = p->next;}return p;
}

平均时间复杂度 O ( n ) O(n) O(n)

求单链表的长度

Length(L):求表长,返回线性表L的长度,即L中数据元素的个数

int Length(LinkList L){int len = 0;		//统计表长LNode *p = L;while(p->next != NULL){p = p->next;len++;}return len;
}

平均时间复杂度 O ( n ) O(n) O(n)

单链表的建立

基于带头结点的单链表讨论

  • Step 1:初始化一个单链表

    typedef struct LNode {          // 定义单链表结点类型ElemType data;              // 每个结点存放一个数据元素struct LNode *next;         // 指针指向下一个结点
    } LNode, *LinkList;// 初始化一个单链表(带头结点)
    bool InitList(LinkList &L){L = (LNode*) malloc(sizeof(LNode));	//分配头结点if (L == NULL) {			//内存不足,分配失败return false;}L->next = NULL;				//头结点后暂时没有存放数据return true;
    }void test(){LinkList L;		//声明一个指向单链表的指针,这里的L是结构体指针//注意此处并没有创建一个结点InitList(L);	// 初始化一个空表//......后续代码....
    }
    
  • Step 2:每次取一个数据元素,插入到表尾/表头

尾插法

思路

初始化单链表
设置变量 length 记录链表长度
while 循环 {每次取一个元素 e;ListInsert(L, length + 1, e) 插到尾部;length++;
}

常规做法

每次取新结点,都从头结点开始遍历到尾部,然后将结点插入到尾部

bool ListInsert(LinkList &L, int i, ElemType e)
{if (i < 1) {return false;}    LNode *p;int j = 0;p = L;// 循环找到第 i-1 个结点while (p != NULL && j < i - 1) {p = p->next;j++;}if (p== NULL) {return false;}LNode *s = (LNode *)malloc(sizeof(LNode));if (s == NULL) {return false;}// 开始插入s->data = e;s->next = p->next;p->next = s;         // 把结点s连到p之后return true;
}

时间复杂度 O ( n 2 ) O(n^2) O(n2)

优化算法

将链表值依次输入,并使用一个临时指针变量记录尾部结点

LinkList List_TailInsert(LinkList &L)
{   // 正向建立单链表int x;                                // ElemType -> 整型L = (LinkList)malloc(sizeof(LNode));  // 头结点if (L == NULL) {return NULL;}LNode *s, *r = L;                      // r为表尾指针while(scanf("%d", &x) != EOF){s = (LNode*)malloc(sizeof(LNode));if (s == NULL) {// ... 清理内存代码...return NULL;}s->data = x;r->next = s;r = s;}r->next = NULL;return L;
}

时间复杂度 O ( n ) O(n) O(n)

头插法

思路

初始化单链表
while (没有取完) {每次取一个数据元素e;ListInsertNode(L,e);
}
LinkList List_HeadInsert(LinkList &L)
{LNode *s;int x;L = (Linklist)malloc(sizeof(LNode));    // 创建头结点if (L == NULL) {return NULL;}L->next = NULL;			//初始为空链表,清除脏数据while (scanf("%d",&x) != EOF) {s = (LNode*)malloc(sizeof(LNode));if (s == NULL) {// ... 清理内存代码...return NULL;}s->data = x;s->next = L->next;L->next = s;}return L;
}

参考文章
【DS 数据结构】003 | 单链表


http://www.ppmy.cn/devtools/15589.html

相关文章

洗地机什么牌子比较好?4款洗地机品牌型号深度推荐

随着科技的不断发展&#xff0c;清洁工具也在不断进化。手持洗地机作为一种新型的清洁工具&#xff0c;因其便捷、高效的特点受到了消费者的青睐。然而&#xff0c;市场上的洗地机品牌众多&#xff0c;消费者在选择时常常感到困惑。那么&#xff0c;哪些洗地机品牌在口碑上表现…

AI大模型与函数式编程

将AI大型模型与函数式编程融合&#xff0c;是一种激动人心的前景。设计模式是解决特定问题的可重复解决方案&#xff0c;它们可以提高代码的可读性、可维护性和可扩展性。而AI大型模型的出现为我们提供了更加智能的解决方案&#xff0c;能够理解和生成自然语言&#xff0c;从而…

146.LRU缓存

题目&#xff1a; 请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构。 实现 LRUCache 类&#xff1a; LRUCache(int capacity) 以 正整数 作为容量 capacity 初始化 LRU 缓存int get(int key) 如果关键字 key 存在于缓存中&#xff0c;则返回关键字的值&…

python - ExcelWriter.book 无法设置属性 ‘book‘

问题描述 conda 环境使用python编辑excel&#xff0c;安装pandas依赖版本为2.2.1。 pandas2.2.1 以下代码片段报错&#xff1a; AttributeError: property book of OpenpyxlWriter object has no setter&#xff08;无法设置属性 book &#xff09; with pd.ExcelWriter(t…

C++ 之 string类 详细讲解

喜欢的人有点难追怎么办 那就直接拉黑 七个女生在一起是七仙女&#xff0c;那七个男生在一起是什么&#xff1f; 葫芦七兄弟 目录 一、为什么要学习string类 二、标准库中的string类 1.string类 2.string类的常用接口说明 2.1 string类对象的常见构造 2.2 string类对…

19-ESP32-S3外设IIC

ESP32-S3的IIC 引言 ESP32-S3是一款集成了Wi-Fi和蓝牙功能的低成本、多功能微控制器。在这篇博客中&#xff0c;我们将详细介绍ESP32-S3的IIC&#xff08;Inter-Integrated Circuit&#xff09;接口&#xff0c;也被称为I2C。 IIC简介 IIC是一种串行、同步、多设备、半双工…

桐乡上元——UI设计

上元教育&#xff0c;创办于2005年&#xff0c;15年一路走来&#xff0c;专注于更优质的课程&#xff0c;只为给学生更好地教学服务&#xff0c;UI设计精英班&#xff0c;历时多年&#xff0c;数次变革&#xff0c;只为塑造UI设计精英&#xff0c;成就学员优质薪资梦想&#xf…

artifactory配置docker本地存储库

​一、概述 本地 Docker 存储库是我们部署和托管内部 Docker 镜像的位置。实际上&#xff0c;它是一个 Docker 注册表&#xff0c;能够托管的 Docker 镜像的集合。通过本地存储库&#xff0c;你可以保存、加载、共享和管理自己的 Docker 镜像&#xff0c;而无需依赖于外部的镜像…