第二章 线性表

线性表

  • 线性表的基本概念
  • 线性表的顺序存储
    • 线性表顺序存储的类型定义
    • 线性表基本运算在顺序表上的实现
    • 顺序表实现算法的分析
  • 线性表的链接存储
    • 单链表的类型定义
    • 线性表的基本运算在单链表上的实现
  • 其他运算在单链表上的实现
    • 建表
    • 删除重复结点
  • 其他链表
    • 循环链表
    • 双向循环链表
  • 顺序实现与链接实现的比较
  • 小试牛刀

线性表的基本概念

  • 线性表:是一种线性结构,由n(n>=0)个数据元素组成的有序序列,数组元素称为结点,n称为表长
  • 线性表通常可表示为(a1,a2,…,an),a1称为起始结点,an称为终端结点。对任意一对相邻结点ai和·ai+1(1<=i<n),ai称为ai+1的直接前驱,ai+1称为ai的直接后继
  • 基本特征:结点具有一对一的关系,结点数不为零,则除起始结点没有直接前驱外,其他每个结点有且仅有一个直接前驱;除终端结点没有直接后继外,其他结点有且仅有一个直接后继
  • 线性表的基本运算及其功能描述
    • 初始化Initiate(L):建立一个空表L=(),L不含数据元素
    • 求表长Length(L):返回线性表L的长度
    • 读表元素Get(L,i):返回线性表第i个数据元素,当i不满足1<=i<=Lenght(L)时,返回一特殊值
    • 定位Locate(L,x):查找线性表中数据元素等于x的数据结点序号,若有多个,则取第一个
    • 插入Insert(L,x,i):在线性表L的第i个元素之前插入一个值为x的新数据元素,表长度加1
    • 删除 Delete(L,i):删除线性表L的第i个数据元素ai,表长度减1

线性表的顺序存储

线性表顺序存储的类型定义

  • 线性表存储的方法:将表中结点依次存放在计算机内存中一组连续的存储单元中
  • 用顺序存储实现的线性表称为顺序表

线性表基本运算在顺序表上的实现

  • 插入:在i处插入x,即ai——an向后移一位,将x置于i,表长+1;算法描述如下:
void InsertSeqList L,DataType x,int i)
{
if (L.length==Maxsize) exit("表已满")
if (i<1 || i>L.length+1) exit("位置错")
for(j=L.length;j>=i;j--)  //从后往前一个一个挪L.data[i-1]=x;L.length++;
}

图解如下:
在这里插入图片描述

  • 删除:删除第i个元素,表长减1
void DeleteSeqList(SeqList L,int i)
{
if(i<1||i>L.length)exit("非法位置")
for(j=i;j<L.length;j++)L.data[j-1]=L.data[j];
L.length--;
}

图示如下:
在这里插入图片描述

  • 定位:查找线性表中值等于x结点序号的最小值,找不到返回0
void LocateSeqlist(Seqlist L,DataType x)
{
int i=0;
while((i<L.length)&&(L.data[i]!=x))i++;
if(i<L.length) return i+1
else return 0;
}

顺序表实现算法的分析

  • 插入算法最坏时间复杂度为O(n),平均时间复杂度为O(n)
  • 删除算法最坏时间复杂度为O(n),平均时间复杂度为O(n)
  • 定位算法最坏时间复杂度为O(n),平均时间复杂度为O(n)
  • 求表长和读表元素算法时间复杂度均为O(1)

线性表的链接存储

单链表的类型定义

  • 单链表——线性表的数据元素用指针链接起来的存储结构,指针表示数据元素之间的逻辑关系,各个结点在内存中的存储位置并不一定连续

在这里插入图片描述
注:单链表可以比作火车,有一个火车头(头指针变量),该变量的值是指向单链表的第一个结点的指针。判断单链表是否为空指针的条件如下:head——>next==NULL或head——>next!=NULL
在这里插入图片描述

线性表的基本运算在单链表上的实现

  • 初始化——创建一个头指针并将其指针域设为NULL,即创建一个空单链表
LinkList InitiateLinkList()
{
LinkList head;	//头指针
head=malloc(sizeof(Node));	//动态构建一结点,为头结点
headhead->next=NULL;
return head;
}
  • 求表长——设计一个工作指针p,初始指向头结点,并设置一个计数器cnt,初值设为0,p每移动一个结点cnt加1,直到p->next==NULL
int LengthLinklist(LinkList head)
{
Node *p=head;
int cnt=0;
while(p->next!=NULL)
{p=p->next;cnt++;
}
return cnt;
}
  • 读表元素——从头开始直到找到给定序号下的元素
Node * GetLinklist(LinklList head,int i)
{
Node *p;
p=head->next;;
int c=1;
while ((c<i)&&(p!=NULL))
{p=p->next;c++;}
if(i==c) return p;
else return NULL;
}
  • 定位——给出值,找到该元素位置(按值查找)
int LocateLinklist(LinkList head,DataType x)
{
Node *p=head;
p=p->next;
int i=0;
while((p!=NULL)&&(p->data!=x))
{
i++;
p=p->next;
}
if(p!=NULL) return i+1;
else return 0;
}
  • 插入——值为x的元素插入到第i个结点之前

在这里插入图片描述
步骤如下:

1.q指针指向i-1结点,p指针指向待加入结点x
2.p指针指向q的直接后继:p->next=q->next;
3. q指针指向p:q->next=p;

void InsertLinklist(LinkList head,DataType x,int i)
{
Node *p,*q;
if(i==1) q=head;;
else q=GetLinklist(head,i-1);
if(q==NULL)exit("找不到插入位置")
else{p=malloc(sizeof(Node));p->data=x;p->next=q->next;q->next=p;}
}
  • 删除:将第i个结点删除

在这里插入图片描述

void DeleteLinklist(LinkList head,int i)
{
Node *p;
if(i==1)q=head;
else q=GetLinklist(head,i-1); //找到待删除结点的直接前驱
if(q!=NULL&& q->next!=NULL)
{
p=q-next;
q->next=p->next;
free(p); //释放已经移出结点p的空间
}
else exit("找不到要删除的结点")
}

其他运算在单链表上的实现

建表

  • 通过插入算法加入新结点
LinkList CreatLinklist1(){
Linklist head;
int x,i;
head=InitiateLinklist();	//建立空表
i=1;
scanf("%d",&x)
while(x!=0);
{
InsertLinklist(head,x,i);
i++;
scanf("%d",&x);		//读下一元素
}
return head;
}

时间复杂度为O(n2

  • 通过一个指向尾结点的指针,将新结点插入到表尾
LinkList CreateLinklist2()
{
Linklist head;
Node *q,*t;
int x;
head=malloc(sizeof(Node))	//生成头结点
q=head;
scanf("%d",%x);
while(x!=0)
{
t=malloc(sizeof(Node));t->data=x;	//生成一个新结点
q->next=t;	//新结点t链入
q=t //修改尾指针q,指向新的尾结点
scanf("%d",&x);
}
q->next=NULL;return head; //q指向尾结点,置尾结点结束
}

在这里插入图片描述
时间复杂度为O(n)

  • 始终将新增加的结点插入到头结点之后
LinkList CreateLinklist3()
{
Linklist head;
Node *p;
int x;
head=malloc(sizeof(Node));	//生成头结点
head->next=NULL;
scanf("%d",&x);
while(x)
{
p=malloc(sizeof(Node));
p->data=x;
p->next=head->next;	//前插,插入到头结点之后第一个结点之前
head->next=p;
scanf("%d",&x);
}
return head;
}

在这里插入图片描述

时间复杂度为O(n)

删除重复结点

  • 一个指针用来确定和谁比较,一个指针移动使得每一项都与之比较(永远指向待删结点的直接前驱),一个指针用于删除
void PurgeLinklist(LinkList head)
{
Node *p,*q,*r
q=head->next;	//q指向首结点
while(q!=NULL){p=q;	//p指向*qwhile(p->next!=NULL)if(p->next->data==q->data) 	//若重复{r=p->next;	//r指向待删除结点p->next=r->next;  //p的直接后继等于r的直接后继,移出待删结点free(r);	//释放}else p=p->next;		//检查下一个q=q->next;	//更新检查结点}
}

在这里插入图片描述

其他链表

循环链表

尾结点的指针域指向第一个结点即构成循环链表

在这里插入图片描述

双向循环链表

在单链表的每个结点中再设置一个指向其直接前驱结点的指针域prior,即为双向循环链表

在这里插入图片描述

  • 双向循环链表的对称性可以用下列等式表示:
p=p->next=p->next->prior
  • 删除(设置一个指针p指向待删结点,待删结点的前驱指向p的直接后继,待删结点的后继指向p的前驱;均由P表示)
p->next->prior=p->next;
p->next->prior=p->prior;
free(p);

在这里插入图片描述

  • 插入
t->prior=p;
t->next=p->next;
p->next->prior=t;
p->next=t;

在这里插入图片描述

顺序实现与链接实现的比较

  • 对于按位置查找,顺序表时间复杂度为O(1),单链表是O(n)
  • 对于定位运算,时间复杂度均为O(n)
  • 对于插入,删除运算,顺序表链表单链表平均时间复杂度均为O(n)
  • 单链表每个结点包括数据域和指针域,指针域需要占用额外空间
  • 顺序表需要预分配存储空间,过大浪费,过小上溢;单链表不用预先分配空间

小试牛刀

  • 设r指向单链表的最后一个结点,要在最后一个结点之后插入s所指的结点,需要执行的语句序列为
______;
r=s;
r->next=NULL;
  • 在单链表中,指针p所致结点为最后一个结点的条件是_____;带头结点的双向循环链表L为空的条件是_____。
  • 在双向循环链表中,在指针p所指结点前插入指针s所指的结点,需要执行下列语句:
s->next=p;
s->prior=p->prior;
p->prior=s;
_______=s。
  • 从逻辑关系 来看,一个数据元素的直接前驱为0个或1个的数据结构只能是______。
  • 单链表中,增加头结点的目的是为了______。

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

如何将图片存到数据库(以mysql为例), 使用ORM Bee更加简单

如何将图片存到数据库 1. 创建数据库: 2. 生成Javabean public class ImageExam implements Serializable {private static final long serialVersionUID 1596686274309L;private Integer id;private String name; // private Blob image;private InputStream image; //将In…

【算法练习Day12】树的递归遍历非递归遍历

​&#x1f4dd;个人主页&#xff1a;Sherry的成长之路 &#x1f3e0;学习社区&#xff1a;Sherry的成长之路&#xff08;个人社区&#xff09; &#x1f4d6;专栏链接&#xff1a;练题 &#x1f3af;长路漫漫浩浩&#xff0c;万事皆有期待 文章目录 递归遍历前序遍历中序遍历后…

《计算机视觉中的多视图几何》笔记(12)

12 Structure Computation 本章讲述如何在已知基本矩阵 F F F和两幅图像中若干对对应点 x ↔ x ′ x \leftrightarrow x x↔x′的情况下计算三维空间点 X X X的位置。 文章目录 12 Structure Computation12.1 Problem statement12.2 Linear triangulation methods12.3 Geomet…

AndroidStudio精品插件集

官网 项目地址&#xff1a;Github博客地址&#xff1a;Studio 精品插件推荐 使用需知 所有插件在 Android Studio 2022.3.1.18&#xff08;长颈鹿&#xff09;上测试均没有问题&#xff0c;推荐使用此版本Android Studio 2022.3.1.18&#xff08;长颈鹿&#xff09;正式版下…

计算机网络(六):应用层

参考引用 计算机网络微课堂-湖科大教书匠计算机网络&#xff08;第7版&#xff09;-谢希仁 1. 应用层概述 应用层是计算机网络体系结构的最顶层&#xff0c;是设计和建立计算机网络的最终目的&#xff0c;也是计算机网络中发展最快的部分 早期基于文本的应用 (电子邮件、远程登…

【计算机网络】HTTPS协议详解

文章目录 一、HTTPS协议 介绍 1、1 HTTP协议不安全的体现 1、2 什么是 HTTPS协议 二、加密的一些概念 2、1 怎么理解加密 2、2 为什么要加密 2、3 常见的加密方式 2、2、1 对称加密 2、2、2 非对称加密 三、HTTPS协议探究加密过程 3、1 只使用对称加密 3、2 只是用非对称加密 3…

JVM篇---第三篇

系列文章目录 文章目录 系列文章目录一、什么是Java虚拟机?为什么Java被称作是“平台无关的编程语言”?二、Java内存结构三、说说对象分配规则一、什么是Java虚拟机?为什么Java被称作是“平台无关的编程语言”? Java虚拟机是一个可以执行Java字节码的虚拟机进程。Java源文…

23.3 Bootstrap 框架4

1. 轮播 1.1 轮播样式 在Bootstrap 5中, 创建轮播(Carousel)的相关类名及其介绍: * 1. carousel: 轮播容器的类名, 用于标识一个轮播组件. * 2. slide: 切换图片的过渡和动画效果. * 3. carousel-inner: 轮播项容器的类名, 用于包含轮播项(轮播图底下椭圆点, 轮播的过程可以显…

【Docker】搭建 Docker 镜像仓库

文章目录 前言&#xff1a;公有仓库和私有仓库公共镜像仓库私有镜像仓库 一、搭建 Docker 镜像仓库1.1 搭建简化版的镜像仓库1.2 搭建带有图形化界面的镜像仓库1.3 配置 Docker 信任地址 二、向私有镜像仓库推送和拉取镜像2.1 推送本地镜像到私有仓库2.2 拉取私有仓库中的镜像 …

机器学习笔记(二)

过拟合 如下图左边,模型出现了过拟合现象 为了解决过拟合现象, 其中一个做法是多收集数据,如右图。 第二种做法是减少模型的特征数量,即x 第三种做法是正则化 正则化就是减少x前面的参数 w的数值, 不用消除x 正则化的梯度下降如下, 因为只是缩小了w的值,而 b的值保持不变 …

通过BeanFactotyPostProcessor动态修改@FeignClient的path

最近项目有个需求&#xff0c;要在启动后&#xff0c;动态修改FeignClient的请求路径&#xff0c;网上找到的基本都是在FeignClient里使用${…}&#xff0c;通过配置文件来定义Feign的接口路径&#xff0c;这并不能满足我们的需求 由于某些特殊原因&#xff0c;我们的每个接口…

floyd算法细节

这个不是一篇学习性文章 主要是针对这几天思考的问题进行一些回答 floyD在计网和数据结构和图模型中有广泛的应用算法 很简单但是其中蕴含的原理值得细究。 弗洛伊德算法(Floyd)主要针对多源最短路径,且可以解决路径中有负权的情况(不包含负权回路),但是迪杰斯特拉算法只…

uni-app:实现页面效果3

效果 代码 <template><view><!-- 风速风向检测器--><view class"content_position"><view class"content"><view class"SN"><view class"SN_title">设备1</view><view class&quo…

【新的小主机】向日葵远程控制ubuntu

向日葵远程控制ubuntu 一、简介二、问题及解决方法2.1 向日葵远程连接Ubuntu22主机黑屏&#xff1f;2.2 Ubuntu如何向日葵开机自启&#xff1f;2.3 无显示器情况下&#xff0c;windows远程桌面连接Ubuntu? 三、美化桌面3.1 安装/解压3.2 设置3.3 右上角显示实时网速 四、安装d…

IBT机考-PBT笔考,优劣分析,柯桥口语学习,韩语入门,topik考级韩语

IBT机考&#xff0c;顾名思义就是在电脑上答题考试&#xff0c;区别于现在的PBT纸笔答题&#xff0c;不需要发卷、收卷&#xff0c;也不需要填涂和用笔写字。 考试不需要带任何文具&#xff0c;就连笔试要用到的修正带都将省去。因为听力、阅读的选择题都是用鼠标点击&#xf…

SpringCloud Alibaba - Seata 四种分布式事务解决方案(XA、AT)+ 实践部署(上)

目录 一、Seata 分布式事务解决方案 1.1、XA 模式 1.1.1、XA模式理论 第一阶段&#xff1a; 第二阶段&#xff1a; 1.1.2、Seata 框架中的 XA 模式 第一阶段&#xff1a; 第二阶段&#xff1a; 1.1.3、XA 模式的优缺点 1.2.4、实现Seata 的 XA 模式 a&#xff09;修改…

FFmpeg:打印音/视频信息(Meta信息)

多媒体文件基本概念 多媒体文件其实是个容器在容器里面有很多流(Stream/Track)每种流是由不同的编码器编码的从流中读出的数据称为包在一个包中包含着一个或多个帧 几个重要的结构体 AVFormatContextAVStreamAVPacket FFmpeg操作流数据的基本步骤 打印音/视频信息(Meta信息…

idea Springboot 教师标识管理系统开发mysql数据库web结构java编程计算机网页源码maven项目

一、源码特点 springboot 教师标识管理系统是一套完善的信息系统&#xff0c;结合springboot框架和bootstrap完成本系统&#xff0c;对理解JSP java编程开发语言有帮助系统采用springboot框架&#xff08;MVC模式开发&#xff09;&#xff0c;系统 具有完整的源代码和数据库&…

无状态自动配置 DHCPv6无状态配置 DHCPv6有状态配置

1、无状态自动配置 配置命令 AR1 ipv6 #开启路由器ipv6报文转发功能 interface GigabitEthernet0/0/0 ipv6 enable #开启路由器接口IPv6报文转发功能 ipv6 address FC01::1/64 …

【C++】AVL树 红黑树

AVL树 AVL树也是二叉搜索树的一种。因为对于普通的二叉搜索树&#xff0c;当插入的数据在有序或接近有序的情况下&#xff0c;二叉搜索树很可能退化成单支树&#xff0c;导致查找效率低下。而AVL树就很好的解决了这个问题。 首先&#xff0c;AVL树是一棵二叉搜索树。同时对于A…