数据结构---线性表

1,顺序表实现---动态分配

#include<stdlib.h>
#define InitSize 10
typedef struct {int *data;//静态分配int length;int MaxSize;
}SqList;
void InitList(SqList& L)
{L.data = (int*)malloc(InitSize * sizeof(int));//分配空间L.length = 0;L.MaxSize = InitSize;//初始空间为10}
void ListInsert(SqList& L, int x, int a)
{L.data[x] = a;
}
void Increaselength(SqList& L, int len)
{int *p = L.data;L.data = (int*)malloc((InitSize + len) * sizeof(int));//重新开辟空间for (int i = 0; i < L.MaxSize;i++){L.data[i] = p[i];}L.MaxSize = InitSize + len;free(p);}
int main()
{SqList q;InitList(q);ListInsert(q, 2, 4);Increaselength(q, 5);ListInsert(q, 12, 2);printf("data[2]=%d,data[12]=%d", q.data[2],  q.data[12]);system("pause");return 0;
}

 第i个元素中的i表示的是位序

2, 静态分配的插入与删除操作:

2.1插入操作:

typedef struct {int data[MaxSize];int length;}SqList;
void InitList(SqList& L)
{L.length = 0;for (int i = 0; i <MaxSize-1; i++){L.data[i] = i;//赋予原始数据,注意不要插满L.length++;}}
bool ListInsert(SqList &L,int i,int e)//在第i个位置插入e
{if (i<1 || i>L.length + 1)return false;if (L.length >= MaxSize)//存储空间已满{return false;}for(int j=L.length;j>=i;j--){L.data[j] = L.data[j - 1];}L.data[i - 1] = e;L.length++;return true;}
int main()
{SqList L;InitList(L);if (ListInsert(L, 8, 1)){printf("data[7]=%d ", L.data[7]);//验证插入}system("pause");return 0;}

2.2,删除操作:

typedef struct {int data[MaxSize];int length;}SqList;
void InitList(SqList& L)
{L.length = 0;for (int i = 0; i < MaxSize - 1; i++){L.data[i] = i;//赋予原始数据,注意不要插满L.length++;}}
bool ListDelete(SqList &L,int i,int &e)//在第i个位置插入e
{if (i<1 || i>L.length)return false;e=L.data[i - 1];//赋值for(int j=i;j<L.length;j++){L.data[j-1] = L.data[j];}L.length--;return true;}int main()
{SqList L;InitList(L);int e = -1;if (ListDelete(L, 8, e))//位置8的元素是7{printf("删除的元素是%d data[7]=%d", e,L.data[7]);//验证删除,删除后data[7]=8}system("pause");return 0;}

3,顺序表的查找:

3.1按位查找

动态分配和普通数组访问方式相同,时间复杂度为O(1)

GetElem(SqList L,int i)
{
return L.data[i-1];
}

3.2按值查找

动态分配方式:

int LocatElem(SqList L, int e)
{
for(int i=0;i<L.length;i++)
{
if(L.data[i]==e)
return i+1;
}
return 0;}

4,链表初始化

4.1不带头结点的单链表的初始化:

与顺序表不同,单链表是用指针来定义

typedef struct LNode {int data;struct LNode* next;//每个元素包括数据和指向下一个节点的指针}LNode,*LinkList;//LNode*等价于LinkList
bool InitList(LinkList &L)//声明是一个单链表
{L = NULL;return true;}
bool IsEmpty(LinkList &L)//声明是一个单链表
{return (L == NULL);}
int main()
{LinkList L;//声明一个指向单链表的指针InitList(L);if(IsEmpty(L))printf("初始化成功");system("pause");return 0;}

4.2,带头结点的单链表的初始化: 

typedef struct LNode {int data;struct LNode* next;//每个元素包括数据和指向下一个节点的指针}LNode,*LinkList;//LNode*等价于LinkList
bool InitList(LinkList &L)//声明是一个单链表
{L = (LNode*)malloc(sizeof(LNode));//头指针指向下一个节点:即为头节点if (L == NULL)return false;//没有内存,分配失败L->next = NULL;//头节点暂时没有节点return true;}
bool IsEmpty(LinkList &L)//声明是一个单链表
{return (L == NULL);}
int main()
{LinkList L;InitList(L);if(IsEmpty(L)==false)printf("初始化成功");system("pause");return 0;}

5,链表插入操作:

5.1,带头结点,按位序插入:

typedef struct LNode {int data;struct LNode* next;//每个元素包括数据和指向下一个节点的指针}LNode,*LinkList;//LNode*等价于LinkList
//带头结点按位序插入
bool ListInsert(LinkList& L,int i,int e)//在第i个位置插入e
{if (i < 1)return false;int j = 0;LNode *p;p= L;//是从第0个节点开始遍历,所以p应该与头节点相同while (p != NULL && j < i - 1)//找到第i-1的节点{p = p->next;j++;}if (p == NULL)//超出链表长度{return false;}LNode *s = (LNode*)malloc(sizeof(LNode));s->data = e;s->next = p->next;p->next = s;return true;}
bool InitList(LinkList &L)//声明是一个单链表
{L = (LNode*)malloc(sizeof(LNode));//头指针指向下一个节点:即为头节点if (L == NULL)return false;//没有内存,分配失败L->next = NULL;//头节点暂时没有节点return true;}
bool IsEmpty(LinkList &L)//声明是一个单链表
{return (L == NULL);}
int main()
{LinkList L;InitList(L);if(IsEmpty(L)==false)printf("初始化成功");if(ListInsert(L, 1, 1))printf("插入成功");system("pause");return 0;}

 5.2,不带头节点按位序插入:

bool ListInsert(LinkList& L,int i,int e)//在第i个位置插入e
{if (i < 1)return false;if (i == 1)//此处插入第一个节点和其他节点不同,直接与L交互{LNode* s = (LNode*)malloc(sizeof(LNode));s->data = e;s->next = L;//指向L指向的节点L = s;//需要更改头结点的指向,比较麻烦return true;}int j = 1;//没有第0个节点LNode *p;p= L;//是从第0个节点开始遍历,所以p应该与头节点相同while (p != NULL && j < i - 1)//找到第i-1的节点{p = p->next;j++;}if (p == NULL)//超出链表长度{return false;}LNode* s = (LNode*)malloc(sizeof(LNode));s->data = e;s->next = p->next;p->next = s;return true;}

5.3,指定节点的后插操作:

//在指定节点后插入一个元素
bool InsertNextLNode(LNode *p,int 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;
}

 

5.4前插操作:

在p节点之前插入s节点

思路:无法找到前驱节点,因此可以插在p节点之后,然后将两个节点中的数据调换

bool InsertPriorNode(LNode* p, LNode* s)
{if (p == NULL || s == NULL)return false;s->next = p->next;p->next = s;int temp = s->data;s->data = p->data;p->data = temp;return true;
}

 6,链表删除操作:

6.1,带头结点按位序删除第i个节点:

思路:先找到第i-1节点,然后将前一个结点的next指针指向后一个结点的next指针

bool ListDelete(LinkList L, int i, int& e)
{if (i < 1)return false;LNode* p = L;//当前指向结点int j = 0;while (p != NULL&&j<i-1) {//找到i-1结点p = p->next;j++;}if (p == NULL)return false;if (p->next== NULL)//需要删除的结点也为空return false;LNode* q = p->next;//应当被删除的结点e = q->data;p->next = q->next;free(q);return true;
}

6.2,指定节点的删除p:

将p的后一个结点复制到p,然后将p的后一个结点释放掉

bool DeleteNode(LNode* p)
{if (p == NULL)return false;LNode* q = p->next;p->data = p->next->data;p->next = q->next;free(q);return true;
}

注意:如果p是最后一个结点,只能从表头依次寻找p的前驱

7,单链表的查找(带头结点)

7.1按位查找

LNode *GetElem2(LinkList L, int i)
{if (i < 0)//第位序0返回的是头节点return NULL;LNode* p;p = L;int j = 0;//带头结点下标从0开始while (p != NULL && j < i){p = p->next;j++;}return p;
}

回顾:

 

7.2按值查找

LNode* GetElem3(LinkList L, int e)
{LNode* p=L->next;while (p != NULL && p->data!=e)//因为头节点中没有数据,所以应该从头节点的下一个结点开始查找{p = p->next;}return p;
}

求表长:

int GetLength(LinkList L)
{LNode* p = L; int len = 0;while (p->next != NULL){p = p->next;len++;}return len;
}

 8,单链表的建立:

8.1尾插法建立单链表

LinkList List_TailInsert(LinkList& L)
{int a = 0;L = (LinkList)malloc(sizeof(LNode));//建立头结点scanf("%d", &a);LNode* r,*s = L;//r是尾结点while (a != 9999)//结束标志{s = (LNode*)malloc(sizeof(LNode));//放入新节点s->data = a;r->next = s;//建立链表连接r = s;//新的尾结点scanf("%d", &a);//循环输入数字}r->next = NULL;return L;}

8.2头插法建立单链表

LinkList List_HeadInsert(LinkList& L)
{int a = 0;LNode* s;L = (LinkList)malloc(sizeof(LNode));//建立头结点L->next = NULL;//初始化单链表,以防有脏数据scanf("%d", &a);while (a != 9999)//结束标志{s = (LNode*)malloc(sizeof(LNode));//放入新节点s->data = a;s->next= L->next;//建立链表连接L->next = s;//新的尾结点scanf("%d", &a);//循环输入数字}return L;}

 尾插法元素排列为顺序,头插法元素排列为逆序,头插法可以用于链表的逆置

9,双链表(带头结点):

9.1定义:

typedef struct DNode {int data;struct DNode *prior,* next;
}DNode,*DLinkList;
bool InitDLinkList(DLinkList &L)
{L = (DNode*)malloc(sizeof(DNode));//分配头结点if (L == NULL)return false;L->prior = NULL;//头结点的前驱节点都为NULL;L->next = NULL;return true;
}
int main()
{DLinkList L;InitDLinkList(L);}

9.2插入操作(在p结点后插入s结点)

bool InsertNextDNode(DNode* p, DNode* s)
{if (p == NULL && s == NULL)return false;s->next = p->next;if (p->next != NULL)//p有后继节点p->next->prior = s;//则后继节点的前继指针指向ss->prior = p;p->next = s;return true;
}

9.3删除操作(删除p结点的后继节点):

bool DeleteNextDNode(DNode* p)
{if (p == NULL)return false;DNode* q = p->next;if (q == NULL)return false;//没有后继节点p->next = q->next;if (q->next != NULL)//后继结点不为最后一个结点q->next->prior = p;//则后继节点的前继指针为sfree(q);return true;
}

9.4销毁双链表:

void DestoryList(DLinkList &L)//L代表的是链表的头指针
{while (L->next != NULL){DeleteNextDNode(L);//从表头开始删除}free(L);//释放头结点L = NULL;//头指针指向NULL
}

9.5遍历:

向后遍历:

while(p!=NULL)
{
p=p->next;
}

向前遍历(跳过头结点):

while(p->prior!=NULL)
{
p=p->prior;}

 

10,循环链表:

10.1循环单链表

定义:

typedef struct LNode {int data;struct LNode* next;
}LNode,*LinkList;//初始化相同
bool InitList(LinkList& L)
{L = (LNode*)malloc(sizeof(LNode));//定义头结点if (L == NULL)//空间分配失败return false;L->next = L;//头指针只向自己return true;
}
bool IsEmpty(LinkList L)
{if (L->next == L)return true;elsereturn false;
}
bool IsTail(LinkList L,LNode *p)//判断p结点是否为尾结点
{if (p->next == L)return true;elsereturn false;
}

从一个结点出发,不同于单链表只能找到后续的各个节点,循环单链表可以找到其他任何一个结点 

10.2循环双链表:

初始化:

判断是否为空是否是表尾结点与循环单链表相同,初始化与双链表不同的是:

L->prior=L;
L->next=L;

插入后继结点:

bool InsertNextDNode(DNode* p, DNode* s)
{s->next = p->next;p->next->prior = s;//双链表需要判断是否是最后一个结点,循环双链表不需要s->prior = p;p->next = s;return true;
}

删除后继结点:

void DeleteNextDNode(DNode* p)
{p->next = q->next;q->next->prior = p;//双链表需要判断是否是最后一个结点,循环双链表不需要free(q);}

11,静态链表:用数组的方式实现链表

初始化:

#define MaxSize 10
typedef struct{
int data;
int next;
}SLinkList[MaxSize];
int main(){
SLinkList a;}

12,顺序表与链表的比较 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/818292.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于SpringCloud,你了解多少?

Why SpringCloud&#xff1f; Spring cloud 是一系列框架的有序集合。它利用 spring boot 的开发便利性巧妙地简化了分布式系统基础设施的开发&#xff0c;如服务发现注册、配置中心、消息总线、负载均衡、断路器、数据监控等&#xff0c;都可以用 spring boot 的开发风格做到一…

论坛直击|发展新质生产力,高校怎么做?

新质生产力浪潮涌动 三大议题聚焦高校人才培养 今年全国两会的政府工作报告将“大力推进现代化产业体系建设&#xff0c;加快发展新质生产力”列在2024年政府工作任务首位&#xff0c;发展新质生产力的先导是培养拔尖创新人才&#xff0c;高等教育改革必须以立德树人为根本任…

带你从BIOS 小工到年薪百万 之 sense amplifier 的作用以及MRC 如何初始化它

做过BIOS 的同学&#xff0c;肯定看过这张图片 对世界充满好奇心的你&#xff0c;是否想过 其中 Sense amplifier 是做什么用的&#xff1f; 是如何工作的&#xff1f; BIOS 又是如何训练它的&#xff1f; 明天接着写

分布式的计算框架之Spark(python第三方库视角学习PySpark)

基本介绍 Apache Spark是专为大规模数据处理而设计的快速通用的计算引擎 。现在形成一个高速发展应用广泛的生态系统。 特点介绍 Spark 主要有三个特点&#xff1a; 首先&#xff0c;高级 API 剥离了对集群本身的关注&#xff0c;Spark 应用开发者可以专注于应用所要做的计…

transformer在生物基因DNA的应用:DNABERT、DNABERT-2

参考&#xff1a; https://www.youtube.com/watch?vmk-Se29QPBA&t1388s 写明这些训练模型可以最终训练好可以进行DNA特征向量的提取&#xff0c;应用与后续1、DNABERT https://github.com/jerryji1993/DNABERT 主要思路就是把DNA序列当成连续文本数据&#xff0c;直接用…

ES11-12

1-ES11-Promise.allSettled Promise.allSettled0)方法返回一个在所有给定的promise都已经fulfilled或rejected后的promise,并带有一个对象数组,每个对象表示对应的promise结果。 简单来说不管成功失败都会调用.then()&#xff0c;然后处理成功和失败的结果 const promises [ …

项目4-图书管理系统2+统一功能处理

1. 拦截器&#xff08;Interceptor&#xff09; 我们完成了强制登录的功能, 后端程序根据Session来判断用户是否登录, 但是实现⽅法是比较麻烦的。 所需要处理的内容&#xff1a; • 需要修改每个接⼝的处理逻辑 • 需要修改每个接⼝的返回结果 • 接⼝定义修改, 前端代码也需…

淘宝商品详情API数据采集接口|如何快速采集淘宝商品数据?

如何快速采集淘宝商品数据 无论是谁&#xff0c;如果单凭人工的方式去收集淘宝、天猫等平台的商品数据信息&#xff0c;工作量是巨大的&#xff0c;如果借助有采集软件的第三方公司操作&#xff0c;则可实现对大数据的轻松掌握&#xff0c;但是外包给第三方公司需要支付一定的…

javaee初阶———多线程(三)

T04BF &#x1f44b;专栏: 算法|JAVA|MySQL|C语言 &#x1faf5; 小比特 大梦想 此篇文章与大家分享多线程专题第三篇,关于线程安全方面的内容 如果有不足的或者错误的请您指出! 目录 八、线程安全问题(重点)1.一个典型的线程不安全的例子2.出现线程不安全的原因3.解决线程不安…

对于普通人来说ChatGPT3.5和4.0的区别,要不要升级GPT4.0

ChatGPT3.5和4.0在官方给出的有哪些区别呢&#xff1f;简单罗列一下&#xff0c;我个人觉得官方给的都太高级&#xff0c;我们平时普通人很难问ChatGPT这种问题。 官方测试所涉及的能力&#xff1a; 视觉能力代码能力数学计算能力工具使用能力与人的交互能力人类专业考试的能…

Node.js从基础到高级运用】二十三、Node.js中自动重启服务器

引言 在Node.js开发过程中&#xff0c;我们经常需要修改代码后重启服务器来应用这些更改。手动重启不仅效率低下&#xff0c;而且会打断开发流程。幸运的是&#xff0c;有一些工具可以帮助我们自动化这个过程。本文将介绍如何使用nodemon来实现Node.js服务器的自动重启。 什么是…

AR智能眼镜方案_MTK平台安卓主板芯片|光学解决方案

AR眼镜作为一种引人注目的创新产品&#xff0c;其芯片、显示屏和光学方案是决定整机成本和性能的关键因素。在这篇文章中&#xff0c;我们将探讨AR眼镜的关键技术&#xff0c;并介绍一种高性能的AR眼镜方案&#xff0c;旨在为用户带来卓越的体验。 AR眼镜的芯片选型至关重要。一…

数据结构---绪论

一、绪论&#xff1a; 1.什么是数据&#xff1f; 数据是信息的载体&#xff0c;是描述客观事物属性的数&#xff0c;字符及所有能输入到计算机中并被计算机程序识别和处理的符号的集合。数据是计算机程序加工的原料。 数据元素--描述一个个体 数据元素&#xff0c;数据项&am…

React + 项目(从基础到实战) -- 第七期

使用ant design 表单组件,开发登录,注册,搜索功能 React 表单组件 ,受控组件 案列 使用defaultVlue属性 bug : 改变了数据源,但是页面未重新渲染 {/* 表单组件 */}<button onClick{()>{console.log(text);}}>打印</button><button onClick{()>[setText(&…

【InternLM 实战营第二期笔记01】书生·浦语大模型全链路开源体系+InternLM2技术报告

本次课程链接在GitHub上&#xff1a;InternLM/Tutorial at camp2 (github.com) 第一次课程录播链接&#xff1a;书生浦语大模型全链路开源体系_哔哩哔哩_bilibili InternLM2技术报告&#xff1a;arxiv.org/pdf/2403.17297.pdf 一、书生浦语大模型全链路开源体系笔记 Intern…

【读书笔记】自动驾驶与机器人中的SLAM技术——高翔

文章会对本书第五章节及以后章节进行总结概括。每日更新一部分。一起读书吧。 第五章——基础点云处理 重点&#xff1a;点云的相邻关系是许多算法的基础 5.1 激光雷达传感器与点云的数学模型 5.1.1激光雷达传感器的数学模型 雷达有两种&#xff1a;机械旋转式激光雷达&…

python 海龟画图tutle螺旋线

目录 初识turtle模块 基本绘图概念 示例&#xff1a;绘制一个正方形 示例&#xff1a;绘制彩色螺旋线 附录 常用命令 其它命令 在Python编程中&#xff0c;使用turtle模块进行图形绘制是一种非常有趣和富有教育意义的活动。通过控制一个小海龟&#xff08;Turtle&#x…

【产品经理修炼之道】- 厂商银业务之保兑仓

保兑仓 保兑仓是指供应商、购货商、银行签订三方协议&#xff0c;以银行信用为载体&#xff0c;以银行承兑汇票为结算工具&#xff0c;由银行控制货权&#xff0c;供应商受托保管货物并对银行承兑汇票保证金以外部分以货物回购为担保措施&#xff0c;购货商随缴保证金随提货而设…

《QT实用小工具·二十六》运行时间记录

1、概述 源码放在文章末尾 运行时间记录&#xff0c;包含如下功能&#xff1a; 可以启动和停止服务&#xff0c;在需要的时候启动。 可以指定日志文件存放目录。 可以指定时间日志输出间隔。 可以单独追加一条记录到日志文件。 日志为文本格式&#xff0c;清晰明了。 软…

AAAI24 - Model Reuse Tutorial

前言 如果你对这篇文章感兴趣&#xff0c;可以点击「【访客必读 - 指引页】一文囊括主页内所有高质量博客」&#xff0c;查看完整博客分类与对应链接。 该篇 Tutorial 主要对 Model Reuse 当下的进展进行了整理和总结。 The Paradigm Shifts Tutorial 中指出在一些数据量比较…