2024重生之回溯数据结构与算法系列学习之顺序表【无论是王道考研人还真爱粉都能包会的;不然别给我家鸽鸽丢脸好嘛?】

本文涉及的产品
实时数仓Hologres,5000CU*H 100GB 3个月
智能开放搜索 OpenSearch行业算法版,1GB 20LCU 1个月
实时计算 Flink 版,5000CU*H 3个月
简介: 顺序表的定义和基本操作之插入;删除;按值查找;按位查找等具体详解步骤以及举例说明

欢迎各位彦祖与热巴畅游本人专栏与博客

你的三连是我最大的动力

以下图片仅代表专栏特色 [点击箭头指向的专栏名即可闪现]

专栏跑道一

➡️网络空间安全——全栈前沿技术持续深入学习

image.gif 编辑

专栏跑道二

➡️ 24 Network Security -LJS

image.gif 编辑

image.gif 编辑

image.gif 编辑

专栏跑道三


➡️ MYSQL REDIS Advance operation

image.gif 编辑

专栏跑道四

➡️HCIP;H3C-SE;CCIP——LJS[华为、华三、思科高级网络]

image.gif 编辑

专栏跑道五

➡️RHCE-LJS[Linux高端骚操作实战篇]

image.png

专栏跑道六

➡️数据结构与算法[考研+实际工作应用+C程序设计]

image.gif 编辑

专栏跑道七

➡️RHCSA-LJS[Linux初级及进阶骚技能]

image.gif 编辑

image.gif

上节回顾



 目录

数据结构王道第2章之顺序表

顺序表的定义和基本操作

定义:

基本操作:

基本操作:

编辑

顺序表的实现-静态分配编辑

 

顺序表的静态分配初始化

如果“数组”存满了怎么办:

顺序表的实现-动态分配:

编辑

顺序表的动态分配初始化代码

顺序表的特点

顺序表的插入删除

顺序表的基本操作-插入:

编辑

增加i的合法性判断:编辑

顺序表的基本操作——删除编辑

插入和删除的时间复杂度:

顺序表的查找

顺序表的按位查找:编辑

顺序表的按位查找代码

顺序表的按值查找:编辑

顺序表的按值查找代码

结构类型的比较

课后习题精选:

(13):给定一个含n个整数的数组Q,请设计一个在时间上尽可能高效的算法,找出数组中未出现的最小正整数

编辑

解题思路

具体代码实现:

(12):已知一个整数序列A=(a0,a1,an-1),其中若存在则称x为A的主元素。

解题思路:

具体代码实现:

(11):一个长度为L的升序序列S,处在第L/2个位置的数称为S的中位数,例如,若序列则S1的中位数是15,两个序列的中位数是11,现在有两个等长升序A和B

解题思路

具体代码实现

(10) :一个长度为L的升序序列S,处在第L/2个位置的数称为S的中位数,例如,若序列则S1的中位数是15,两个序列的中位数是11,现在有两个等长升序A和B

编辑

解题思路:

具体代码实现:


数据结构王道第2章之顺序表

顺序表的定义和基本操作

定义:

基本操作:

  • 用顺序存储的方式实现线性表顺序存储,把逻辑上相邻的元素存储在物理位置上也相邻的存储单元中,元素之间的关系由存储单元的邻接关系来体现

基本操作:

  • InitList(&L):初始化表。构造一个空的线性表L,分配内存空间
  • DestroyList(&L):销毁操作。销毁线性表,并释放线性表L所占用的内存空间
  • ListInsert(&L,i,e):插入操作。在表L中的第i个位置上插入指定元素e
  • ListDelete(&L,i,&e):删除操作。删除表L中第i个位置的元素,并用e返回删除元素的值
  • LocateElem(L,e):按值查找操作。在表L中查找具有给定关键字值的元素
  • GetElem(L,i):按位查找操作。获取表L中第i个位置的元素的值
  • Length(L):求表长。返回线性表L的长度,即L中数据元素的个数
  • PrintList(L):输出操作。按前后顺序输出线性表L的所有元素值
  • Empty(L):判空操作。若L为空表,则返回true,否则返回false
  • 什么时候要传入参数的引用“&”对参数的修改结果需要“带回来”,是引用类型而不是值类型

image.gif

顺序表的实现-静态分配 image.gif

image.gif 编辑

顺序表的静态分配初始化

#define MaxSize 10      // 定义最大长度
typedef struct{
    ElemType data[MaxSize];     // 用静态的“数组”存放数据元素
    int length;     // 顺序表的当前长度
}SqList;
image.gif
#include <stdio.h>
#define MaxSize 10
typedef struct{
    int data[MaxSize];
    int length;
}SqList;
void InitList(SqList &L)
{
    // 可以省略,但可能由于遍历时用到MaxSize有脏数据,要用length遍历
    for (int i = 0; i < MaxSize; i ++ )
        L.data[i] = 0;
    
    L.length = 0;       // 不可省略,顺序表初始长度为0
}
int main()
{
    SqList L;       // 声明一个顺序表
    InitList(L);        // 初始化顺序表
    
    return 0;
}
image.gif

如果“数组”存满了怎么办:

可以放弃治疗,顺序表的表长刚开始确定后就无法更改(存储空间是静态的),同时如果提前初始化太多的空间而不用,又会造成资源的浪费,因此动态分配应运而生。

动态申请和释放内存空间:

  • C:malloc、free函数
  • L.data = (ElemType *) malloc (sizeof(ElemType) * InitSize);
  • malloc函数返回一个指针, 空间需要强制转型为你定义的数据元素类型指针
  • malloc函数的参数指明要分配多大的连续内存空间
  • C++: new、delete关键字

顺序表的实现-动态分配:

image.gif

顺序表的实现:

  • 随机访问,即可以在O(1)时间内找到第i个元素。
  • 存储密度高,每个结点只存储数据元素
  • 拓展容量不方便(即便采用动态分配的方式实现,拓展长度的时间复杂度也比较高)
  • 插入、删除操作不方便,需要移动大量元素

顺序表的动态分配初始化代码

#define InitSize 10     // 顺序表的初始长度
typedef struct
{
    ElemType *data;     // 指示动态分配数组的指针,这个指针指向顺序表中第一个数据元素
    int MaxSize;        // 顺序表的最大容量
    int length;     // 顺序表的当前长度
} SeqList;      // 顺序表的类型定义(动态分配方式)
image.gif
// 动态申请和释放空间
// 在C语言中的函数分别是malloc和free函数
// malloc函数是申请一整片连续的内存空间,且会return一个指向这一整片存储空间开始地址的指针,需要强制转型为你定义的数据元素类型的指针
// L.data = (ElemType *)malloc(sizeof(ElemType) * InitSize);
// malloc和free包含在<stdlib.h>头文件中
// 在C++语言中分别是new和delete这两个关键字
image.gif
  • 这样就可以让顺序表的容量可变
  • 虽然动态分配可以使顺序表的大小可以灵活改变,但是时间开销还是比较大的(复制元素)
  • 注意malloc和free是一对函数
  • free函数会把p这个指针所指向的这一整片的存储空间给释放掉,归还给系统,然后由于p是局部于这个函数的变量,函数结束后,存储p这个变量的存储空间会被系统自动回收

image.gif 编辑

#include <stdlib.h>
#define InitSize 10
typedef struct
{
    int *data;
    int MaxSize;
    int length;
} SeqList;
void InitList(SeqList &L)
{
    L.data = (int *)malloc(sizeof(int) * InitSize);
    L.MaxSize = InitSize;
    L.length = 0;
}
void IncreaseSize(SeqList &L, int len)
{
    int *p = L.data;
    L.data = (int *)malloc(sizeof(int) * (L.MaxSize + len));
    for (int i = 0; i < L.length; i ++ )
        L.data[i] = p[i];
    L.MaxSize = L.MaxSize + len;
    free(p);
}
int main()
{
    SeqList L;
    InitList(L);
    // ...往顺序表中随意随便插入几个元素
    IncreaseSize(L, 5);
    return 0;
}
image.gif

顺序表的特点

  • 随机访问,即可以在O ( 1 ) O(1)O(1)时间内找到第i个元素(不论是静态分配还是动态分配代码都是d a t a [ i − 1 ] data[i - 1]data[i−1]
  • 存储密度高,每个节点只存储数据元素(链表还要存指针)
  • 拓展容量不方便(即便采用动态分配的方式实现,拓展长度的时间复杂度也比较高)
  • 插入、删除操作不方便,需要移动大量元素

image.gif

顺序表的插入删除

顺序表的基本操作-插入:

image.gif

  • ListInsert(&L, i, e) :插入操作,在表L中的第i个位置(位序)插入指定元素i
  • 本节代码建立在顺序表的“静态分配”实现方式之上,“动态分配”也雷同。
  • 时间复杂度的平均情况 :p = 1 / ( n + 1 ) p=1/(n+1)p=1/(n+1);i=1,循环n次,i=2,循环n-1次…;平均循环次数= n p + ( n − 1 ) ∗ p + . . . + 1. p = n ∗ ( n + 1 ) / 2 ∗ 1 / ( n + 1 ) = n / 2 =np+(n-1)*p+...+1.p=n*(n+1)/2*1/(n+1)=n/2=np+(n−1)∗p+...+1.p=n∗(n+1)/2∗1/(n+1)=n/2
#define MaxSize 10
typedef struct
{
    int data[MaxSize];
    int length;
} SqList;
bool ListInsert(SqList &L, int i, int e)
{
    if (i < 1 || i > L.length + 1)      // 判断i的范围是否有效
        return false;
    if (L.length == MaxSize)        // 当前存储空间已满,不能插入
        return false;
    for (int j = L.length; j >= i; j -- )       // 将第i个元素及之后的元素后移
        L.data[j] = L.data[j - 1];
    L.data[i - 1] = e;      // 在位置i处放e
    L.length ++ ;       // 长度加1
    
    return true;        // 反馈
}
int main()
{
    SqList L;
    InitList(L);
    // ...插入一些元素
    ListInsert(L, 5, 5);
    
    return 0;
}
  • image.gif

增加i的合法性判断: image.gif

顺序表的基本操作——删除 image.gif

bool ListDelete(SqList &L, int i, int &e)
{
    if (i < 1 || i > L.length)      // 判断i的范围是否有效
        return false;
    
    e = L.data[i - 1];      // 将被删除的元素赋给e
    
    for (int j = i; j < L.length; j ++ )        // 将第i个位置后的元素前移
        L.data[j - 1] = L.data[j];
    
    L.length -- ;     // 线性表长度减一
    
    return true;
}
int main()
{
    SqList L;
    InitList(L);
    // ...插入一些元素
    
    int e = -1;     // 用变量e把删除的元素“带回来”
    
    if (ListDelete(L, 3, e))
        printf("已删除第3个元素,删除元素值为=%d\n", e);
    else
        printf("位序i不合法,删除失败");
}
image.gif
  • ListDelete(&L, i, &e) :删除操作,删除表L中第i个位置的元素,并用e返回删除元素的值
  • 因为要返回e,所以这要有一个引用操作,因此,在这个函数中操作的变量e,在内存中其实对应的是同一份数据
  • 在删除操作中是先移动前面的元素再移动后面的元素,而在插入操作中要把元素往后移时,先把后面的元素往后移,然后再移前面的元素

插入和删除的时间复杂度:

  • 最好时间复杂度= O(1)
  • 最坏时间复杂度= O(n)
  • 平均时间复杂度= O(n)

image.gif

顺序表的查找

顺序表的按位查找: image.gif

顺序表的按位查找代码

// 静态分配实现顺序表,动态分配实现的顺序表也是如此
ElemType GetElem(SqList L, int i)
{
    // 判断i合法性
    
    return L.data[i - 1];
}
image.gif
  • GetElem(L, i) :按位查找操作。获取表L中第i个位置的元素的值
  • 正是如此,在初始化顺序表时候malloc需要强制转换为与数据元素的数据类型相对应的指针
  • 时间复杂度= O(1)
  • 随机存取:由于顺序表的各个数据元素在内存中连续存放,因此可以根据起始地址和数据元素大小立即找到第i个元素,

顺序表的按值查找: image.gif

顺序表的按值查找代码

#define InitSize 10
typedef struct
{
    ElemType *data;
    int MaxSize;
    int length;
} SeqList;
ElemType LocateElem(SeqList L, ElemType e)
{
    for (int i = 0; i < L.length; i ++ )
        if (L.data[i] == e)
            return i + 1;       // 返回位序
    return 0;
}
image.gif
  • LocateElem(L, e) :按值查找操作,在表L中查找具有给定关键字值的元素
  • 结构类型的数据元素也能用 == 比较吗:不能!(C++可以用 == 的重载来实现)
  • 更好的办法:定义一个函数
  • 依次对比各个分量来判断两个结构体是否相等
  • 最好时间复杂度= O(1)
  • 最坏时间复杂度= O(n)
  • 平均时间复杂度= O(n)

结构类型的比较

  • 在C语言中,结构类型的比较不能直接用“==”,需要依次对比各个分量来判断两个结构体是否相等;如果C++,则可以用重载 “= =“。
  • 但是,《数据结构》考研初试中,手写代码可以直接用“= =”,无论ElemType是基本数据类型还是结构类型。但是有的学校考《C语言程序设计》,那么也许语言就要严格一些。最好还是看一下相关的历年真题。






相关文章
|
3天前
|
弹性计算 双11 开发者
阿里云ECS“99套餐”再升级!双11一站式满足全年算力需求
11月1日,阿里云弹性计算ECS双11活动全面开启,在延续火爆的云服务器“99套餐”外,CPU、GPU及容器等算力产品均迎来了全年最低价。同时,阿里云全新推出简捷版控制台ECS Lite及专属宝塔面板,大幅降低企业和开发者使用ECS云服务器门槛。
|
21天前
|
存储 弹性计算 人工智能
阿里云弹性计算_通用计算专场精华概览 | 2024云栖大会回顾
阿里云弹性计算产品线、存储产品线产品负责人Alex Chen(陈起鲲)及团队内多位专家,和中国电子技术标准化研究院云计算标准负责人陈行、北京望石智慧科技有限公司首席架构师王晓满两位嘉宾,一同带来了题为《通用计算新品发布与行业实践》的专场Session。本次专场内容包括阿里云弹性计算全新发布的产品家族、阿里云第 9 代 ECS 企业级实例、CIPU 2.0技术解读、E-HPC+超算融合、倚天云原生算力解析等内容,并发布了国内首个云超算国家标准。
阿里云弹性计算_通用计算专场精华概览 | 2024云栖大会回顾
|
3天前
|
人工智能 弹性计算 文字识别
基于阿里云文档智能和RAG快速构建企业"第二大脑"
在数字化转型的背景下,企业面临海量文档管理的挑战。传统的文档管理方式效率低下,难以满足业务需求。阿里云推出的文档智能(Document Mind)与检索增强生成(RAG)技术,通过自动化解析和智能检索,极大地提升了文档管理的效率和信息利用的价值。本文介绍了如何利用阿里云的解决方案,快速构建企业专属的“第二大脑”,助力企业在竞争中占据优势。
|
1天前
|
人工智能 自然语言处理 安全
创新不设限,灵码赋新能:通义灵码新功能深度评测
自从2023年通义灵码发布以来,这款基于阿里云通义大模型的AI编码助手迅速成为开发者心中的“明星产品”。它不仅为个人开发者提供强大支持,还帮助企业团队提升研发效率,推动软件开发行业的创新发展。本文将深入探讨通义灵码最新版本的三大新功能:@workspace、@terminal 和 #team docs,分享这些功能如何在实际工作中提高效率的具体案例。
|
7天前
|
负载均衡 算法 网络安全
阿里云WoSign SSL证书申请指南_沃通SSL技术文档
阿里云平台WoSign品牌SSL证书是由阿里云合作伙伴沃通CA提供,上线阿里云平台以来,成为阿里云平台热销的国产品牌证书产品,用户在阿里云平台https://www.aliyun.com/product/cas 可直接下单购买WoSign SSL证书,快捷部署到阿里云产品中。
1849 6
阿里云WoSign SSL证书申请指南_沃通SSL技术文档
|
10天前
|
Web App开发 算法 安全
什么是阿里云WoSign SSL证书?_沃通SSL技术文档
WoSign品牌SSL证书由阿里云平台SSL证书合作伙伴沃通CA提供,上线阿里云平台以来,成为阿里云平台热销的国产品牌证书产品。
1789 2
|
19天前
|
编解码 Java 程序员
写代码还有专业的编程显示器?
写代码已经十个年头了, 一直都是习惯直接用一台Mac电脑写代码 偶尔接一个显示器, 但是可能因为公司配的显示器不怎么样, 还要接转接头 搞得桌面杂乱无章,分辨率也低,感觉屏幕还是Mac自带的看着舒服
|
26天前
|
存储 人工智能 缓存
AI助理直击要害,从繁复中提炼精华——使用CDN加速访问OSS存储的图片
本案例介绍如何利用AI助理快速实现OSS存储的图片接入CDN,以加速图片访问。通过AI助理提炼关键操作步骤,避免在复杂文档中寻找解决方案。主要步骤包括开通CDN、添加加速域名、配置CNAME等。实测显示,接入CDN后图片加载时间显著缩短,验证了加速效果。此方法大幅提高了操作效率,降低了学习成本。
5386 15
|
13天前
|
人工智能 关系型数据库 Serverless
1024,致开发者们——希望和你一起用技术人独有的方式,庆祝你的主场
阿里云开发者社区推出“1024·云上见”程序员节专题活动,包括云上实操、开发者测评和征文三个分会场,提供14个实操活动、3个解决方案、3 个产品方案的测评及征文比赛,旨在帮助开发者提升技能、分享经验,共筑技术梦想。
1139 152
|
21天前
|
存储 缓存 关系型数据库
MySQL事务日志-Redo Log工作原理分析
事务的隔离性和原子性分别通过锁和事务日志实现,而持久性则依赖于事务日志中的`Redo Log`。在MySQL中,`Redo Log`确保已提交事务的数据能持久保存,即使系统崩溃也能通过重做日志恢复数据。其工作原理是记录数据在内存中的更改,待事务提交时写入磁盘。此外,`Redo Log`采用简单的物理日志格式和高效的顺序IO,确保快速提交。通过不同的落盘策略,可在性能和安全性之间做出权衡。
1585 14