01-线性表 (数据结构和算法)

news2025/1/13 15:42:25

要点:

  • 程序 = 数据结构 + 算法

一、数据结构的概述

程序 = 数据结构 + 算法

数据结构:计算机存储、组织数据的方式

算法:处理数据的方式

1.1 基本概念和术语

1、数据

数据(data):所有能够输入到计算机中去的描述客观事物的符号

  • 数值型数据: 表示数量,由数字、小数点、正负号和表示乘幂的字母E组成。数值型的数据是不能包含文本的,必须是数值

  • 非数值型数据:如文字、图像、声音等的计算机应用领域

2、数据元素

数据元素(data element):数据的基本单位,也称之为结点(node)或者记录(record)

3、数据对象

数据对象(data object):相同特性数据元素的集合,是数据的一个子集

  • 整型数据对象:所有正负整数的集合,其中的每一个整数就是一个数据元素

1.2 数据结构的分类

传统上,我们可以把数据结构分为逻辑结构物理结构两大类:

  • 逻辑结构分类:逻辑结构是从具体问题中抽象出来的模型,是抽象意义上的结构,按照对象中数据元素之间的相互关系分类,也是我们后面课题中需要关注和讨论的问题

    • 集合结构:集合结构中数据元素除了属于同一个集合外,他们之间没有任何其他的关系

    • 线性结构:线性结构中的数据元素之间存在一对一的关系

    • 树形结构:树形结构中的数据元素之间存在一对多的层次关系

    • 图形结构:图形结构的数据元素是多对多的关系

  • 物理结构分类:逻辑结构在计算机中真正的表示方式(又称为映像)称为物理结构(实际存储结构),也可以叫做存储结构。常见的物理结构有顺序存储结构、链式存储结构。
    • 顺序存储结构把数据元素放到地址连续的存储单元里面,其数据间的逻辑关系和物理关系是一致的 ,比如我们常用的数组就是顺序存储结构。

    • 链式存储结构把数据元素存放在任意的存储单元里面,这组存储单元可以是连续的也可以是不连续的。此时,数据元素之间并不能反映元素间的逻辑关系,因此在链式存储结构中引进了一个指针存放数据元素的地址,这样通过地址就可以找到相关联数据元素的位置。

二、线性表

2.1 线性结构概述

1、线性表是一种典型的线性结构:则有且仅有一个开始结点和一个终端结点,并且所有结点都最多只有一个直接前趋和一个直接后继

2、线性结构表达式:(a1, a2, a3, ......  , ai)

3、线性结构特点:

  • 只有一个首结点和尾结点

  • 除首尾结点外,其他结点只有一个直接前驱和一个直接后继

简言之,线性结构反映结点间的逻辑关系是一对一的。

4、线性结构包括线性表、堆栈、队列、字符串、数组等等。

线性表是最基本、最简单、也是最常用的一种数据结构。一个线性表是n个具有相同特性的数据元素的有限序列。

如果把线性表用数学语言来定义,则可以表示为(a1,...ai-1,ai,ai+1,...an),ai-1领先于ai,ai领先于ai+1,称ai-1是ai的前驱元素,ai+1是ai的后继元素

线性表中数据存储的方式可以是顺序存储,也可以是链式存储,按照数据的存储方式不同,可以把线性表分为顺序表和链表

线性表的基本操作:初始化、取值、查找、插入、删除。

顺序表是在计算机内存中以数组的形式保存的线性表,线性表的顺序存储是指用一组地址连续的存储单元,依次存储线性表中的各个元素。  

typedef int ElemType;
typedef unsigned int uint;
typedef struct{
    ElemType *elem;  //顺序表中存储数据的空间
    uint length;     //当前存储数据的个数
    uint listSize;   //顺序表的容量,最多可以存储的元素的个数
}SqList;

2.2 初始化顺序表

逻辑:

  • 为存储数据的指针分配空间

  • 顺序表长度初始化为0

  • 顺序表的容量为 分配的空间  sizeof(ElemType)

/*
 * @brief  初始化顺序表
 * @param listSize 顺序表容量
 * @return 返回初始化好的顺序表
 * */
SqList SqList_init(uint listSize){
    //为线性表存储数据分配空间
    SqList t;
    t.elem = (ElemType *)malloc(listSize*sizeof(ElemType));
    t.length = 0;
    t.listSize = listSize;
    return t;
}

2.3 顺序表的输出

逻辑:

  • 根据顺序表的长度遍历整个顺序表

  • 将遍历到的每一个数据元素输出

/*
 * @brief 打印顺序表的所有的元素
 * @param t 需要输出的顺序表
 * @return void
 * */
void print_SqList(SqList t){
    printf("SqList print: ");
    int i;
    for (i = 0; i < t.length; i++){
        printf("SqList_data, %d", t.elem[i]);
    printf("\n");
    }
}

2.4 顺序表的取值

逻辑:

  • 判断需要获取的元素下标是否合法

  • 如果不合法则返回一个出错编号

  • 如果合法直接返回下标所对应的元素

/*
 * @brief 获取顺序表中某个位置的元素
 * @param t 需要操作的顺序表
 * @param index 需要获取的元素的位置/下标
 * @return ElemType 获取到的元素
 * */
ElemType get_elem(SqList t, uint index){
    if (index >= t.length){
        printf("index out of range \n");
        return -1;
    }
    return t.elem[index];
}

2.5 顺序表元素的查找

逻辑:

  • 通过顺序表的长度遍历整个顺序表

  • 将遍历到的元素与需要查找的元素比较,如果不相等则继续往后遍历

  • 如果相等则停止遍历,退出循环

  • 返回遍历到的元素的下标

/* @brief 在顺序表中查找指定的元素
 * @param t 需要操作的顺序表
 * @param elem 需要查找的元素
 * @return int 成功返回值元素的下标,失败返回-1
 * */
int find_elem(SqList t, ElemType elem){
    int index = -1;  //保存元素的下标,默认需要查找的元素不在顺序表中
    //遍历整个顺序表
    int i;
    for (i = 0; i < t.length; i++){
        if (t.elem[i] == elem){
            index = i;
            break;
        }
    }
    
    if (-1 == index)
        printf("Can not find element : %d\n", elem);
    return index;
}

2.6 顺序表的销毁

逻辑:

  • 释放初始化顺序表时在堆上分配的空间

  • 将顺序表的长度设置为0

  • 将顺序表的容量设置为0

/*
 * @brief 销毁一个顺序表
 * @param t 需要销毁的顺序表指针
 * */
int SqList_destroy(SqList *t){
    if (NULL == t)
        return error;
    if (t->elem != NULL){
        free(t -> elem);
        t->length = 0;
        t->listSize = 0;
    }
    return success;
}

2.7 删除指定位置的元素

逻辑:

  • 判断需要删除的元素下标是否合法,如果不合法返回出错码

  • 将第i+1至length-1(最后一个元素的下标)位的元素往前移动一个位置

  • 顺序表长度 -1

/*
 * @brief 删除指定位置的元素
 * @param t 需要操作的顺序表指针
 * @param index 需要删除的元素的下标
 * */
int locate_elem_delete(SqList *t, uint index){
    if (NULL == t)
        return error;
    if (index >= t->length){
        printf("index out of range ...\n");
        return error;
    }
    int i;
    //删除第index个节点,后面的数据移动 t->length - (index + 1)次
    //将第index+1开始到第t.length-1位置的元素往前移动
    for (i=0; i < t->length - (index + 1); i++){
        t->elem[index+i] = t->elem[index+i+1];
    }
    t->length--;
    return success;
}

2.8 删除指定的元素

逻辑:

  • 遍历整个顺序表

  • 判断遍历到得元素是否为需要删除得元素,如果不是则继续往后遍历

  • 如果相等则将该元素后面得所有元素往前移动一个位置

/*
 * @brief 将顺序表中指定的元素删除
 * @param t 需要操作的顺序表指针
 * @param elem 需要删除的元素
 * @return 成功返回OK,失败返回error
 * */
int delete_designated_elem(SqList *t, ElemType elem){
    if (NULL == t){
        printf("[%s %d] SqList is NULL\n", __FUNCTION__ , __LINE__);
        return error;
    }
    
    int i = 0;
    while (i != t->length){
        //先找到需要删除的元素
        if (t->elem[i] != elem){
            i++;
            continue;
        }
        //记录其位置
        int p = i;
        int j;
        //将后面的元素往前移动一个位置
        for (j = 0; j < t->length - (p + 1); j++){
            //将第index + i + 1个元素 覆盖掉第 index + i个元素
            t->elem[p + j] = t->elem[p + j + 1];
        }
        t->length--;
    }
    return success;
}

2.9 顺序表的扩容

逻辑:

  • 将顺序表的容量扩大到原来的两倍

/*
 * @brief 为顺序表扩容
 * @param t 需要扩容的顺序表指针
 * @return 成功返回success, 失败返回error
 * */
int SqList_expand(SqList *t){
    if (NULL == t){
        printf("[%s %d] SqList is NULL \n", __FUNCTION__ , __LINE__);
        return error;
    }
    //为顺序表分配新的空间
    t->listSize *= 2;
    t->elem = (ElemType *)realloc(t->elem, t->listSize * sizeof(ElemType));
    return success;
}

2.10 在指定位置前插入元素

逻辑:

  • 判断插入的位置是否合法

  • 判断顺序表是否满了,如果满了则扩容

  • 根据插入的位置计算需要移动的元素个数/次数

  • 将指定位置开始的所有元素整体往后移动一个位置(从最后一个位置开始往后移动)

  • 将需要插入的元素放置到插入的位置

  • 顺序表长度+1

/*
 * @brief 在指定位置的前面插入一个元素
 * @param t 需要操作的顺序表指针
 * @param index 需要插入的位置
 * @param elem 需要插入的元素
 * @return error 成功返回success, 失败返回error
 * */
int elem_insert(SqList *t, uint index, ElemType elem){
    if (NULL == t){
        printf("[%s %d] SqList is NULL \n", __FUNCTION__ , __LINE__);
        return error;
    }
    //判断插入的位置是否合法
    if (index > t->length)
        return error;
    //判断顺序表是否满了,如果满了则扩容
    if (t->length == t->listSize){
        SqList_expand(t);
        print_SqList(*t);
    }
    int i;
    for (i = 0; i < t->length - index; i++){
        //从最后一个元素开始移动
        t->elem[t->length+i] = t->elem[t->length-1+i];
    }
    //将需要插入的元素放置到插入的位置
    t->elem[index] = elem;
    t->length++;
    return success;
}

2.11 顺序表的时间复杂度

1、获取元素的时间复杂度

因为可以通过下标法直接获取到对应位置的元素,因为不论顺序表的长度是多少多只需要要访问一次就能够获取到元素,因此时间复杂度为O(1)

2、插入元素的时间复杂度

每一次插入,都需要把对应位置后面的元素移动一次,随着元素数量N的增大,移动的元素也越多,时间复杂为O(n)

3、删除元素的时间复杂度

每一次删除,都需要把对应位置后面的元素移动一次,随着数据量N的增大,移动的元素也越多,时间复杂度为O(n)

 

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

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

相关文章

【Method】稀疏与压缩感知 | 图像稀疏性及压缩感知方法白话讲解

【Method】稀疏与压缩感知 | 图像稀疏性及压缩感知方法白话讲解 文章目录 【Method】稀疏与压缩感知 | 图像稀疏性及压缩感知方法白话讲解1. 为什么图像是可压缩的&#xff1a;图像空间的广阔2. 什么是Sparsity&#xff1f;3.压缩感知&#xff1a;简介4.压缩感知&#xff1a;数…

matlab学习指南(3):最全MATLAB工具箱Toolbox下载地址大汇总

&#x1f305;*&#x1f539;** φ(゜▽゜*)♪ **&#x1f539;*&#x1f305; 欢迎来到馒头侠的博客&#xff0c;该类目主要讲数学建模的知识&#xff0c;大家一起学习&#xff0c;联系最后的横幅&#xff01; 喜欢的朋友可以关注下&#xff0c;私信下次更新不迷路&#xff0…

火车头采集器AI伪原创【php源码】

本文介绍火车头采集器AI伪原创&#xff0c;对于新媒体从业者来说&#xff0c;会写文章是最基本的职业技能&#xff0c;而伪原创是我们经常使用的技能。今天我要讲的是SEO标兵如何在伪原创上创作文章。 首先&#xff0c;原创性永远是最好的&#xff0c;更受读者欢迎。伪原创的出…

Microsoft Dynamics 365:VS2019引用BC发布的SOAP服务

1、搜索网页服务 2、点击新建 3、选择对象类型&#xff1a;页面、单元、查询&#xff0c;输入ID&#xff0c;勾选即可发布服务 4、复制SOAP URL到浏览器里看看是否可以访问&#xff0c;这样就OK的 5、 右键添加服务引用 6、选择高级 7、添加web引用 8、服务地址粘贴进去查找服…

云原生监控——VictoriaMetrics

1.简介 VictoriaMetrics是一个快速高效且可扩展的监控解决方案和时序数据库&#xff0c;可以作为Prometheus的长期远端存储&#xff0c;具备的特性有&#xff1a; 支持prometheus查询api&#xff0c;同时实现了一个metricsql 查询语言支持全局查询视图&#xff0c;支持多prom…

uniapp引用leaflet地图实现方案

最近在做uniapp实现的移动端app&#xff0c;其中一些模块需要gis地图&#xff0c;在最开始的时候我尝试了使用uniapp官方自带的map组件&#xff0c;但是非常不好用。 后来又引用了mars2d来实现&#xff0c;但是发现这种引用方式会出现一个bug&#xff0c;在浏览器当中使用的时候…

Blueprint —— 入门笔记

蓝图比C性能较慢&#xff1b; 蓝图起作用需在场景中创建实例&#xff1b; 在Event Graph内 按住右键&#xff0c;平移界面&#xff1b;滚动滚轮&#xff0c;缩放界面&#xff1b;按住左键节点&#xff0c;移动节点&#xff1b;右击&#xff0c;显示节点对话框&#xff1b;按住…

外部存储器接口(EMIF)

1 接口信号与控制寄存器 EMIF(External Memory Interface)外部存储器接口为DSP芯片与众多外部设备之间提供一种连接方式&#xff0c;EMIF最常见的用途就是同时连接FLASH和SDRAM。EMIF性能优良&#xff0c;跟外部SDRAM和异步器件连接时&#xff0c;具有很大的方便性和灵活性。根…

ModaHub魔搭社区:常用的相似性度量——浮点向量相似性度量和二进制向量相似性度量

目录 常用的相似性度量 浮点向量相似性度量 二进制向量相似性度量 总结 常用的相似性度量 如果没有相似性度量——计算两个向量之间距离的方法,再好的向量数据库也没有用。因为存在许多度量,我们在这里只讨论最常用的子集。 浮点向量相似性度量 最常见的浮点向量相似…

【数据分析 - 基础入门之pandas篇①】- pandas介绍

文章目录 前言一、pandas介绍二、pandas优势2.1 强大的数据结构支撑2.2 优点 三、pandas学习路线结语相关导读 前言 一、pandas介绍 pandas 是 Python 的 核心数据分析支持库 &#xff0c;提供了快速、灵活、明确的数据结构&#xff0c;旨在简单、直观地处理关系型、标记型数据…

【JUC进阶】11. BlockingQueue

目录 1、前言 2、BlockingQueue 2.1、ArrayBlockingQueue 2.1.1、take() 2.1.2、put() 2.2、LinkedBlockingQueue 2.3、PriorityBlockingQueue 2.4、SynchronousQueue 3、简单使用 3.1、创建ArrayBlockingQueue 3.2、Demo 1、前言 对于并发程序而言&#xff0c;高性…

python: FileHelper

# encoding: utf-8 # 版权所有 2023 涂聚文有限公司 # 许可信息查看&#xff1a; # 描述&#xff1a; # Author : geovindu,Geovin Du 涂聚文. # IDE : PyCharm 2023.1 python 311 # Datetime : 2023/7/9 19:12 # User : geovindu # Product : PyCharm # Proj…

QT事件处理

设计一个闹钟&#xff0c;定时播报内容。 #ifndef MAINWINDOW_H #define MAINWINDOW_H#include <QMainWindow> #include <QTimerEvent> #include <QDateTime> #include <QMessageBox> #include <QTextToSpeech> #include <QDebug> namespa…

校园闲置物品交易平台的设计与实现(论文+源码)_kaic

摘 要 伴随大数据时代的到来&#xff0c;计算机已成为人们步入个数化生活的必须品。由于计算机技术的成熟&#xff0c;互联网的强大功能也正在被人们以最大限度的开发。通过网络&#xff0c;人们能够足不出户完成校园闲置物品查阅&#xff0c;这在方便学生的同时也解决了在传统…

用于FPGA远程更新的QuickBoot方法

用于FPGA远程更新的QuickBoot方法 用于FPGA远程更新的QuickBoot方法 用于FPGA远程更新的QuickBoot方法1. 远程更新简介2 QuickBoot方案2.1 QuickBoot配置方法2.2 QuickBoot Flash 编程方法 3.QuickBoot实现3.1 Critical Switch World (key point)3.2 QuickBoot存储映射3.3 Bits…

Django ORM中QuerySet常用接口汇总记录

存在模型数据 学生表 课程表 支持链式操作的接口 all接口&#xff1a;用于查询所有数据&#xff0c;相当于&#xff1a;select * from xxx filter接口&#xff1a;根据条件过滤数据 values接口&#xff1a;指定返回的字段&#xff0c;结果是包含 dict 的 QuerySet 对象 valu…

zabbix----代理服务器,高可用集群

文章目录 一、部署 zabbix 代理服务器1.1 设置 zabbix 的下载源&#xff0c;安装 zabbix-proxy1.2 部署数据库&#xff0c;要求 MySQL 5.7 或 Mariadb 10.5 及以上版本1.3修改 zabbix-proxy 配置文件1.4 启动 zabbix-proxy1.5 在所有主机上配置 hosts 解析1.6 在 Web 页面配置 …

Oracle批量生成供datax调用的json文件及可执行sh脚本

Oracle+DataX+存储过程实现异构库之间的数据同步资源-CSDN文库 背景: 项目需要做数据迁移(hive2oceanbase),两边的库有几百张表,人工生成json文件,工作量巨大,想来想去还是用Oracle存储过程的形式,批量生成json文件和shell脚本,及实现跑批的功能。 本次测试是Oracl…

免费插画网站

humaaans undraw iradesign fresh-folk delesign

MYSQL的体系结构

mysql体系结构可以分为四个层级&#xff1a;连接层&#xff0c;SQL层&#xff0c;插件存储引擎&#xff0c;物理文件层 补充&#xff1a; SQL层中&#xff1a; 管理服务和工具组件&#xff1a;从备份和恢复的安全性、复制、集群、管理、配置、迁移和元数据等方面管理数据库。…