数据结构——单链表上基本操作的实现

2024-01-20 19:28

本文主要是介绍数据结构——单链表上基本操作的实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.按位序插入(带头结点)

==ListInsert(&L, i, e): ==在表L中的第i个位置上插入指定元素e = 找到第i-1个结点(前驱结点),将新结点插入其后;其中头结点可以看作第0个结点,故i=1时也适用。

typedef struct LNode{ ElemType data; struct LNode *next;

}LNode, *LinkList;

//在第i个位置插入元素e带头结点

bool ListInsert(LinkList &L, int i, ElemType e){

//判断i的合法性, i是位序号(1开始) if(i<1)

LNode *p; int j=0;

p = L;

//循环找到第i-1个结点

while(p!=NULL && j<i-1){ p = p->next;

j++;

}

if (p==NULL)

return false;

//在第i-1个结点后插入新结点

LNode *s = (LNode *)malloc(sizeof(LNode)); //申请一个结点s->data = e;

s->next = p->next;

p->next = s;                   //将结点s连到p,后两步千万不能颠倒qwq

return true;

}

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

2.按位序插入(不带头结点)

==ListInsert(&L, i, e): ==在表L中的第i个位置上插入指定元素e = 找到第i-1个结点(前驱结点),将新结点插入其后; 因为不带头结点,所以不存在0结点,因此!i=1 时,需要特殊处理——插入(删除)1个元素时,需要更改头指针L;

typedef struct LNode{ ElemType data; struct LNode *next;

}LNode, *LinkList;

bool ListInsert(LinkList &L, int i, ElemType e){

if(i<1)

return false;

//插入到第1个位置时的操作有所不同! if(i==1){

LNode *s = (LNode *)malloc(size of(LNode)); s->data =e;

s->next =L;

L=s;           //头指针指向新结点return true;

}

//i>1的情况与带头结点一样!唯一区别是j的初始值为1 LNode *p;  //指针p指向当前扫描到的结点int j=1;    //当前p指向的是第几个结点

p = L;           //L指向头结点,头结点是第0个结点(不存数据)

//循环找到第i-1个结点

while(p!=NULL && j<i-1){ p = p->next;

j++;

}

if (p==NULL)

return false;

//在第i-1个结点后插入新结点

LNode *s = (LNode *)malloc(sizeof(LNode)); //申请一个结点s->data = e;

s->next = p->next;

p->next = s; return true;

}

3.指定结点的后插操作:

==InsertNextNode(LNode *p, ElemType e):== 给定一个结点p,在其之后插入元素e; 根据单链表的链接指针只能往后查找,故给定一个结点p,那么p之后的结点我们都可知,但是p结点之前的结点无法得 ;

typedef struct LNode{ ElemType data; struct LNode *next;

}LNode, *LinkList;

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;}

//有了后插操作,那么在第i个位置上插入指定元素e的代码可以改成:

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++;

}

return InsertNextNode(p, e)

}

4.指定结点的前插操作

思想:设待插入结点是s,将s插入到p的前面。我们仍然可以将s插入到*p的后面。然后将p->datas-

>data交换,这样既能满足了逻辑关系,又能是的时间复杂度为O(1)

//前插操作:在p结点之前插入元素e

bool InsertPriorNode(LNode *p, ElenType e){ if(p==NULL)

return false;

LNode *s = (LNode *)malloc(sizeof(LNode)); if(s==NULL) //内存分配失败

return false;

//重点

s->next = p->next;

p->next = s; //新结点s连到p之后s->data = p->data; //p中元素复制到s p->data = e; //p中元素覆盖为e

return true

}  //时间复杂度为O(1)

5.按位序删除节点(带头结点)

==ListDelete(&L, i, &e):== 删除操作,删除表L中第i个位置的元素,并用e返回删除元素的值;头结点视为

0结点;

思路:找到第i-1个结点,将其指针指向第i+1个结点,并释放第i个结点;

typedef struct LNode{

ElemType data;

struct LNode *next;

}LNode, *LinkList;

bool ListDelete(LinkList &L, int i, ElenType &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;

if(p->next == NULL) //i-1个结点之后已无其他结点

return false;

LNode *q = p->next; e = q->data;

p->next = q->next;

free(q)

return true;

}

时间复杂度分析:

最坏,平均时间复杂度:O(n)

最好时间复杂度:删除第一个结点 O(1)

6.指定结点的删除

bool DeleteNode(LNode *p){ if(p==NULL)

return false;

LNode *q = p->next;       //q指向*p的后继结点

p->data = p->next->data; //p和后继结点交换数据域p->next = q->next;   //*q结点从链中断开” free(q);

return true;

} //时间复杂度 = O(1)

心得体会

1. 链表的动态性质:链表结构可以在运行时动态地插入和删除节点,这是它与数组最大的不同之处。链表不需要预分配固定的存储空间,可以根据需要动态分配。

2. 头结点的便捷性:使用头结点可以简化插入和删除操作,因为无论在链表的任何位置进行这些操作,都有一个统一的节点来参考,即头结点。

3. 指针的重要性:链表的操作很大程度上依赖于指针,正确地移动和更新指针是确保链表结构正确性和稳定性的关键。

4. 复杂度的理解:虽然链表允许O(1)时间复杂度的元素插入和删除(在某些条件下),但按位序操作通常需要O(n)的时间复杂度,因为可能需要遍历整个链表以找到正确的位置。

5. 内存管理:在C中使用链表时,必须小心处理内存的分配和释放。每次创建新节点时,都需要使用`malloc`分配内存,并在删除节点时使用`free`释放内存,以避免内存泄漏。

6. 边界条件的处理:在链表的操作中,需要特别注意边界条件,例如插入或删除第一个元素时,可能需要特殊处理,比如更新头指针。

7. 错误处理:适当的错误处理是链表操作中不可忽视的部分。例如,当内存分配失败时,需要返回错误信息,并避免程序崩溃。

8. 算法优化:有时候,通过一些巧妙的方法可以优化链表的操作,比如前插操作可以通过交换数据来避免复杂的节点断开和连接,这样可以减少一些不必要的指针操作。

这篇关于数据结构——单链表上基本操作的实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/627082

相关文章

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

Java使用Thumbnailator库实现图片处理与压缩功能

《Java使用Thumbnailator库实现图片处理与压缩功能》Thumbnailator是高性能Java图像处理库,支持缩放、旋转、水印添加、裁剪及格式转换,提供易用API和性能优化,适合Web应... 目录1. 图片处理库Thumbnailator介绍2. 基本和指定大小图片缩放功能2.1 图片缩放的

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

PyCharm中配置PyQt的实现步骤

《PyCharm中配置PyQt的实现步骤》PyCharm是JetBrains推出的一款强大的PythonIDE,结合PyQt可以进行pythion高效开发桌面GUI应用程序,本文就来介绍一下PyCha... 目录1. 安装China编程PyQt1.PyQt 核心组件2. 基础 PyQt 应用程序结构3. 使用 Q

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库