数据结构——线性表①(顺序表)

news2024/11/25 13:47:39

一、线性表定义

线性表是一种数据结构,它是由n个具有相同数据类型的数据元素a1,a2,…,an组成的有限序列

其中,除第一个元素a1外,每一个元素有且只有一个直接前驱元素,除了最后一个元素an外,每一个元素有且只有一个直接后继元素。

线性表可以用顺序存储结构链式存储结构来实现。

  • 顺序表是一种用一段地址连续的存储单元依次存储线性表中的数据元素的存储结构;
  • 链表则是一种用一组任意的存储单元存储线性表中的数据元素的存储结构。

【线性表内容框架】
在这里插入图片描述

二、线性表特点

  • 表中数据元素的个数有限
  • 表中元素具有逻辑上的顺序性,表中元素有其先后次序
  • 表中元素都是数据元素,每个元素都是单个元素
  • 表中的元素数据类型都相同,即每个元素都占有相同大小的存储空间
  • 表中元素具有抽象性,即仅讨论元素之间的逻辑关系,而不考虑元素究竟表示什么内容

三、线性表的基本操作

  • 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 。

注意
上面写的函数严格来说并不正确,因为没有写返回值类型
其次,函数的参数有带符号“&”的,这是C++的语法,和C语言中的指针效果一致
比如说初始化表的函数,用C语言的写法,也可以写成 InitList(<类型名>* L);
要初始化一个数据元素类型为整型的线性表的话,初始化函数可以写成 InitList(int* L);

四、线性表的分类

(1)顺序表

1.1 顺序表的定义

线性表的顺序存储叫做顺序表。(用顺序存储的方式实现线性表)

顺序存储——把逻辑上相邻的元素存储在物理位置上也相邻的存储单元中,元素之间的关系由存储单元的邻接关系来体现。需要开辟一段了连续的空间用来存储数据。

在这里插入图片描述

1.2 顺序表的创建和初始化及相关操作

这里插一句,顺序表有两种,一种是动态顺序表,一种是静态顺序表
·静态顺序表使用定长数组存储元素
·而动态顺序表可以动态开辟内存,可以动态改变数组长度

静态顺序表的创建和初始化比较简单,
在这里插入图片描述
静态顺序表的创建就是直接在main函数中创建一个SqList 类型的结构体变量
静态顺序表的初始化就是把 链表结构体中的 length 的值设置为0

静态顺序表的基本操作:添加元素(插入元素)、删除元素、修改元素、查找元素

代码如下:

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>

#define MaxSize 10 //这里宏定义了静态顺序表最多能存储10个数据

struct SqList
{
	int data[MaxSize];
	int length;
};
typedef struct SqList SqList;

void InitList(SqList* list)//初始化静态顺序表
{
	list->length = 0;
}

int ListInsert(SqList* L, int i, int e)//在表中位置i处,插入数据e
{
    if (i < 1 || i > L->length + 1 || L->length >= MaxSize) {
        return 0;
    }
    for (int j = L->length; j >= i; j--) 
    {
        L->data[j] = L->data[j - 1];
    }
    L->data[i - 1] = e;
    L->length++;
    return 1;
}

int ListDelete(SqList* L, int i) //删除表中位序为i的数据元素
{
    if (i < 1 || i > L->length)
    {
        return 0;
    }
    for (int j = i; j < L->length; j++)
    {
        L->data[j - 1] = L->data[j];
    }
    L->length--;
    return 1;
}

void PrintList(SqList list)//打印顺序表中所有元素
{
    for (int i = 0; i < list.length; i++)
    {
        printf("%d ", list.data[i]);
    }
}

int Find(SqList list,int e)//查找顺序表中值为e 的元素,返回其位序,如果没找到,返回0
{
    if (list.length == 0)
    {
        return 0;
    }
    for (int i = 0; i < list.length; i++)
    {
        if (list.data[i] == e)
        {
            return i + 1;
        }
    }
    return 0;
}

int Change(SqList* list,int i,int e)//修改位序为i处的数据,把数据改成e
{
    if (i < 1 || i > list->length + 1 || list->length >= MaxSize) {
        return 0;
    }
    list->data[i-1] = e;
    return 1;
}
#include"SqList.h"

int main() 
{
    SqList L;
    InitList(&L);
    ListInsert(&L, 1, 10);//在位序1处插入数字10
    ListInsert(&L, 2, 20);//在位序2处插入数字20
    ListInsert(&L, 3, 30);//在位序3处插入数字30
    PrintList(L);//打印静态顺序表中的所有数据
    printf("\n");
    ListDelete(&L, 2);//删除静态顺序表中位序为2的数据(删除20)
    PrintList(L);
    printf("\n");
    int pos1 = Find(L, 20);
    printf("%d\n", pos1);//0
    int pos2 = Find(L, 30);
    printf("%d\n", pos2);//2
    Change(&L,1,24);
    PrintList(L);//24 30
    return 0;
}

在这里插入图片描述


动态顺序表的创建和初始化

#define InitSize 10  //顺序表的初始长度
typedef struct
{
	ElemType *data; //指示动态分配数组的指针
	int MaxSize;    //顺序表的最大容量
	int length;     //顺序表的当前长度
}seqList;
//顺序表的类型定义(动态顺序表)

动态顺序表的好处是可以在容量不够的情况下进行扩容操作
动态顺序表的创建:在main函数中创建一个结构体变量
动态顺序表的初始化:用malloc开辟一段连续空间;初始化最大容量为宏定义的初始化长度;令顺序表的当前长度为0

动态顺序表的增删改查的函数操作实现

代码如下

#define InitSize 10  //顺序表的初始长度  
typedef int ElemType;
typedef struct seqList
{
	ElemType* data; //动态分配数组的指针  
	int MaxSize;    //顺序表的最大容量  
	int length;     //顺序表的当前长度  
} seqList;

void InitList(seqList* L)//动态顺序表的初始化
{
	L->data = (int*)malloc(InitSize * sizeof(int));//用malloc函数申请一片连续的存储空间
	L->length = 0;
	L->MaxSize = InitSize;
}

//增加动态数组的长度
void Increasesize(seqList* L, int len)
{
	int* p = L->data;
	L->data = (int*)malloc((L->MaxSize + len) * sizeof(int));
	for (int i = 0; i < L->length; i++)
	{
		L->data[i] = p[i];
	}
	L->MaxSize = L->MaxSize + len; 
	free(p);
}

// 插入元素  
int InsertList(seqList* L, int index, ElemType elem) {
    if (index < 0 || index > L->length || L->length == L->MaxSize) {
        return 0; // index error 或 溢出  
    }
    for (int i = L->length; i >= index; i--) {
        L->data[i + 1] = L->data[i]; // 后移一位  
    }
    L->data[index] = elem; // 插入新元素  
    L->length++; // 长度加一  
    return 1; // 插入成功  
}

// 删除元素  
int DeleteList(seqList* L, int index) {
    if (index < 0 || index >= L->length) {
        return 0; // index error  
    }
    for (int i = index; i < L->length - 1; i++) {
        L->data[i] = L->data[i + 1]; // 前移一位  
    }
    L->length--; // 长度减一  
    return 1; // 删除成功  
}

// 查找元素,返回元素位置,未找到返回-1  
int FindList(seqList* L, ElemType elem) {
    for (int i = 0; i < L->length; i++) {
        if (L->data[i] == elem) {
            return i; // 找到元素,返回位置  
        }
    }
    return -1; // 未找到元素,返回-1  
}

// 修改元素,返回修改结果,未找到返回-1  
int UpdateList(seqList* L, int index, ElemType elem) {
    if (index < 0 || index >= L->length) {
        return 0; // index error  
    }
    L->data[index] = elem; // 修改元素值  
    return 1; // 修改成功,返回1  
}

1.3 顺序表的特点

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

1.4顺序表的应用场景

顺序表适用于需要随机访问元素的场景,例如需要快速查找某个元素的位置或者根据下标进行访问。

以下是顺序表的一些应用场景举例:

  • 数组:数组是一种顺序表的实现方式,适用于需要快速访问元素的场景,例如存储图像、音频等数据。
  • 数据库:数据库中的表可以使用顺序表来实现,例如需要根据主键快速查找某个记录。
  • 缓存:缓存中的数据可以使用顺序表来实现,例如需要快速查找某个缓存项。
  • 索引:索引可以使用顺序表来实现,例如需要快速查找某个关键字对应的记录。
  • 排序:排序算法中的一些算法可以使用顺序表来实现,例如冒泡排序、快速排序等。

~~《数据结构——线性表②》敬请期待…
在这里插入图片描述

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

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

相关文章

IDEA 如何运行 SpringBoot 项目

步骤一&#xff1a;配置 Maven 第一步&#xff1a;用 IDEA 打开项目&#xff0c;准备配置 maven 环境 &#xff0c;当然如果本地没有提前配置好 maven&#xff0c;就用 IDEA 默认的配置即可 配置 maven 步骤 情况 1&#xff1a;如果本地没有配置过 maven&#xff0c;可以保持如…

【Spring】IOC容器与Bean的常用属性配置

文章目录 1.前言2.IOC容器2.1 BeanFactory 容器2.2 ApplicationContext 容器 3.Bean的常用属性配置4. 总结 1.前言 在之前的文章-IOC的快速入门中讲过Bean这个概念. 本来就来介绍容器与Bean的常用属性配置 在Spring框架中&#xff0c;Bean指的是被Spring加载生成出来的对象。 …

ubuntu 18.04 编译安装flexpart 10.4(2023年) —— 筑梦之路

2023年10月29日 环境说明 操作系统版本&#xff1a;ubuntu 18.04 python版本&#xff1a;3.6.9 gcc版本&#xff1a;7.5.0 编译安装路径&#xff1a;/usr/local cmake: 3.10.2 所需要的源码包我已经打包放到我的资源。 2021年1月份已经写过一篇Ubuntu 编译安装的帖子F…

Android OpenGL ES 2.0入门实践

本文既然是入门实践&#xff0c;就先从简单的2D图形开始&#xff0c;首先&#xff0c;参考两篇官方文档搭建个框架&#xff0c;便于写OpenGL ES相关的代码&#xff1a;构建 OpenGL ES 环境、OpenGL ES 2.0 及更高版本中的投影和相机视图。 先上代码&#xff0c;代码效果如下图…

MQ——进阶

文章目录 消息可靠性生产者消息确认消息持久化消费者确认演示none模式演示auto模式 失败重试机制本地重试失败策略 死信交换机初始死信交换机TTL延迟队列安装DelayExchange插件使用DelayExchange 惰性队列消息堆积问题惰性队列 MQ集群集群分类普通集群镜像模式镜像模式的配置 仲…

私有云:【6】VCenter安装SqlServer

私有云&#xff1a;【6】VCenter安装SqlServer 1、VCenter安装SqlServer1.1、通过模板创建虚拟机1.2、安装sqlserver服务 2、搭建sqlserver群集2.1、安装群集功能2.2、在ad域服务器创建共享文件夹&#xff0c;供集群选举使用 3、创建故障转移群集【只需安装一台即可】3.1、创建…

操作系统 --- 存储器管理

一、简答题 1.存储器管理的基本任务&#xff0c;是为多道程序的并发执行提供良好的存储器环境。请问好的存储器环境”应包含哪几个方面&#xff1f; 答&#xff1a; 2.内存保护是否可以完全由软件实现&#xff1f;为什么&#xff1f; 答&#xff1a;内存保护的主要任务是确保每…

LeetCode热题100——双指针

双指针 1.移动零2.盛最多水的容器3.三数之和 1.移动零 给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 // 题解&#xff1a;使用双指针&#xff0c;其中快指针指向非零元素&#xff0c;慢指针指向首个零元素下…

msvcp120.dll怎么修复,五种方法教你如何修复msvcp120.dll文件

在运行软件时&#xff0c;我们常常会遇到一些错误提示&#xff0c;其中之一就是“由于找不到msvcp120.dll无法继续执行代码”。这个错误通常发生在使用Microsoft Visual C 2013编译的程序运行时。本文将介绍5种修复这个问题的方法&#xff0c;帮助各位解决这个困扰。 方法一、使…

如何使用批量重命名的方法替换重复文件名内容

在文件管理过程中&#xff0c;我们有时会遇到文件名中包含相同部分内容的情况&#xff0c;这不仅会使文件显得混乱&#xff0c;而且还会给文件检索和使用带来不便。为了解决这个问题&#xff0c;我们可以使用云炫文件管理器批量重命名进行批量替换。下面是如何使用这种方法进行…

RT-Thread入门

1、初识RT-Thread RT-Thread&#xff0c;全称是Real Time-Thread&#xff0c;即嵌入式实时多线程操作系统。其基本属性之一是支持多任务&#xff0c;但是允许多任务同时运行&#xff0c;但是并不是意味着处理器在同一时刻真的执行了多个任务。实际上&#xff0c;一个处理器核心…

mac 安装homebrew ,golang

mac 安装homebrew ,golang 安装homebrew安装golang选择 apple arm 版本安装配置环境变量 安装homebrew /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)"回车执行指令后&#xff0c;根据提示操作。具体包括以下提示操作&am…

深度学习(4)---生成式对抗网络(GAN)

文章目录 一、原理讲述1.1 概念讲解1.2 生成模型和判别模型 二、训练过程2.1 训练原理2.2 损失函数 三、应用 一、原理讲述 1.1 概念讲解 1. 生成式对抗网络&#xff08;Generative Adversarial Network&#xff0c;GAN&#xff09;是一种深度学习模型&#xff0c;是近年来复杂…

vue vant van-uploader使用compressorjs解决拍照上传的图片被旋转 90 度方法,图片压缩上传

vue vant van-uploader使用compressorjs解决拍照上传的图片被旋转 90 度方法&#xff0c;图片压缩上传_van-uploader 拍照上传服务器后图片翻转-CSDN博客文章浏览阅读3.2k次&#xff0c;点赞4次&#xff0c;收藏6次。van-uploader使用compressorjs解决拍照上传的图片被旋转 90 …

游戏找不到x3daudio1_7.dll无法继续执行的解决方法,快速解决dll问题

x3daudio1_7.dll是一个音频处理库&#xff0c;主要用于实现三维音频渲染。它包含了微软的XAudio2音频API&#xff0c;该API被许多游戏和应用程序用于实现高质量的音频效果。这个库文件主要处理音频的空间定位、响度均衡以及多通道音频输出等功能。在游戏中&#xff0c;它可以为…

08 MIT线性代数-求解Ax=b:可解性与结构Complete Solution of Ax=b

1. 可解的条件 Solvability conditions on b 检验Axb是否可解的方法是对增广矩阵进行行消元。如果矩阵A的行被完全消去的话&#xff0c;则对应的b的分量也要得0 两条关于b的限制条件(等价) 1. if a comb. of rows of A gives zero row, then same comb. of enties of b must …

笔记46:ResNeXt 网络详解

本地笔记地址&#xff1a;D:\work_file\DeepLearning_Learning\03_个人笔记\2.图像处理任务\ResNeXt网络学习 a a a a a a a a a a a a a a

【论文阅读VLDB23】Online Schema Evolution is (Almost) Free for Snapshot Databases

Online Schema Evolution is (Almost) Free for Snapshot Databases 对snapshot DB来说可以几乎在线进行schema的演变。 ABSTRACT 在现有数据库系统中&#xff0c;对在线和事务模式演变的支持仍然具有挑战性。之前处理这种schema 演变基本是patches补丁的方式来做&#xff0…

实战授权码登录流程

我是经常阅读公众号优质文章&#xff0c;也经常体验到公众号的授权登录功能。今天就来实现下&#xff0c;流程图如下 效果图 后端接口 主要用来接收微信服务器推送的公众号用户触发的事件、生成和验证授权码的有效性 解析微信服务器推送的事件通知 public String login(Se…

深度强化学习用于博弈类游戏-基础测试与说明【1】

深度强化学习用于博弈类游戏-基础【1】 1. 强化学习方法2. 强化学习在LOL中的应⽤2.1 环境搭建2.2 游戏特征元素提取1)小地图人物位置&#xff1a;2)人物血量等信息3)在整个图像上寻找小兵、防御塔的位置4&#xff09;自编码器提取 3. 策略梯度算法简介参考资料 1. 强化学习方法…