【数据结构】单链表

server/2024/9/22 18:35:34/

单链表

文章目录

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

定义

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

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

单链表的优缺点

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

用代码定义单链表

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/server/22582.html

相关文章

[NeurIPS-23] GOHA: Generalizable One-shot 3D Neural Head Avatar

[pdf | proj | code] 本文提出一种基于单图的可驱动虚拟人像重建框架。基于3DMM给粗重建、驱动结果&#xff0c;基于神经辐射场给细粒度平滑结果。 方法 给定源图片I_s和目标图片I_t&#xff0c;希望生成图片I_o具有源图片ID和目标图片表情位姿。本文提出三个分支&#xff1a;…

C语言-atoi和atof函数的使用

人生应该树立目标&#xff0c;否则你的精力会白白浪费。&#x1f493;&#x1f493;&#x1f493; 目录 •&#x1f319;知识回顾 &#x1f34b;知识点一&#xff1a;atoi函数的使用和实现 • &#x1f330;1.函数介绍 • &#x1f330;2.代码演示 • &#x1f330;3.atoi函数的…

002 springboot redis 防止表单重复提交

文章目录 RedisConfig.javaWebConfiguration.javaAutoIdempotentTokenController.javaMyOrderController.javaMyOrder.javaAutoIdempotentInterceptor.javaAutoIdempotentIdempotentTokenService.javaIdempotentTokenServiceImpl.javaSpringbootRedisDemoApplication.javaappli…

【MySQL】1.安装与配置

目录 1.数据库介绍 1.1什么是数据库 1.2数据库分类 2.MySQL服务器安装 2.1Windows绿色安装 2.2Windows中重装MYSQL 3.Mac中常见的安全问题 4.客户端连接MYSQL服务器 5.SQL的分类 1.数据库介绍 1.1什么是数据库 文件保存数据有以下的缺点&#xff1a; 文件的安全性问…

等保测评常见安全风险

在进行等保测评时&#xff0c;需要特别注意的常见安全风险包括但不限于以下几个方面&#xff1a; 1. **信息泄露风险**&#xff1a;评估系统中存储、传输和处理的敏感信息的安全性&#xff0c;防止数据被非法获取&#xff0c;避免个人隐私泄露或公司机密泄露等问题。 2. **拒…

4.Docker本地镜像发布至阿里云仓库、私有仓库、DockerHub

文章目录 0、镜像的生成方法1、本地镜像发布到阿里云仓库2、本地镜像发布到私有仓库3、本地镜像发布到Docker Hub仓库 Docker仓库是集中存放镜像的地方&#xff0c;分为公共仓库和私有仓库。 注册服务器是存放仓库的具体服务器&#xff0c;一个注册服务器上可以有多个仓库&…

uniapp真机调试无法调用之前页面的方法

在uniapp通过getCurrentPages&#xff08;&#xff09;页面栈调用之前页面方法&#xff0c;h5可生效但app真机调试找不到方法 let pages getCurrentPages()let beforePage pages[pages.length - 3]beforePage.refresh() //真机调试refresh为undefined解决&#xff1a; 后面…

【vscode环境配置系列】vscode远程debug配置

VSCODE debug环境配置 插件安装配置文件debug 插件安装 安装C/C, C/C Runner 配置文件 在项目下建立.vscode文件夹&#xff0c;然后分别建立c_cpp_properties.json&#xff0c; launch.json&#xff0c;tasks.json&#xff0c;内容如下&#xff1a; c_cpp_properties.json:…