【C++数据结构】顺序存储结构的抽象实现

news2025/1/6 19:09:34

文章目录

  • 前言
  • 一、目标
  • 二、SeqList实现要点
  • 三、SeqList函数实现
    • 3.1 get函数
    • 3.2 set函数
    • 3.3 insert函数
      • 带2个参数的insert
      • 带一个参数的insert
    • 3.4 remove函数
    • 3.5 clear函数
    • 3.6 下标运算符重载函数
      • 无const版本
      • const版本
    • 3.7 length函数
  • 总结


前言

当谈到C++数据结构时,顺序存储结构是一个重要的概念。它是一种将数据元素按照其逻辑顺序依次存储在内存中的方式。这种存储方式使得元素在内存中是连续存储的,这有助于快速访问和操作数据。在本文中,我们将讨论顺序存储结构的抽象实现,以帮助您更好地理解它的原理和应用。

顺序存储结构是一种基本的数据结构,通常用于线性表(如数组)的表示。它有助于我们在计算机程序中存储和操作一系列数据元素,例如整数、字符、对象等。在顺序存储结构中,元素按照它们的逻辑顺序在内存中依次排列,这种存储方式使得元素的访问非常高效,因为我们可以通过索引快速定位任何元素。


一、目标

本节课要实现承上启下的一个非常重要的类:SeqList,当我们实现了他我们才会去实现我们具体的StaticList和DynamicList,所以他是非常重要的。那么我们上节课也已经介绍了实现流程如果不会的同学可以去看看上节课
在这里插入图片描述

二、SeqList实现要点

SedList 设计要点

  • 抽象类模板,存储空间的位置和大小由子类完成
  • 实现顺序存储结构线性表的关键操作 (增,删,查,等)
  • 提供数组操作符,方便快速获取元素
    那么我们可以抽象出下面这个类
template <typename T>
class SeqList :public List<T>
{
    T* m_array;//存储位置,由子类申请
    int m_length;//数组长度

public:
    bool insert(const T& e);

    bool insert(int i, const T& e);

    bool remove(int i);

    bool set(int i, const T& e);

    bool get(int i, T& e) const;

    int length() const;

    void clear();

    T& operator[](int i);

    T operator[](int i)const;

    virtual int capacity() const = 0;
};

在这里插入图片描述

三、SeqList函数实现

3.1 get函数

顺序存储结构的元素获取操作

  • 判断目标位置是否合法
bool ret = ((0 <= i) && (i < m_length));

在这里插入图片描述

让我们来解释一下:

(0 <= i) 意味着 i 必须大于或等于0,因为在大多数编程中,位置通常从0开始编号,所以不能小于0。

(i < m_length) 意味着 i 必须小于线性表的长度 m_length。这是因为线性表中的位置索引不能超过线性表的总元素个数。

这两个条件一起的意思是,我们需要确保 i 不仅大于等于0,还要小于线性表的长度,这样我们才能访问线性表中有效的位置。如果 i 不满足这两个条件,那么访问它可能导致越界错误,这是一种非常常见的编程错误,因此需要进行这样的合法性检查,以防止程序出现问题。

  • 将目标位置作为数组下标获取元素
    因为我们这是原生数组,所以直接使用下标操作即可
e = m_array[i];

总体代码如下:

bool get(int i, T& e) const
{
    bool ret = ((0 <= i) && (i < m_length));

    if (ret)
    {
        e = m_array[i];
    }

    return ret;
}

在这里插入图片描述

3.2 set函数

顺序存储结构的元素设置操作

  • 判断目标位置是否合法
bool ret = ((0 <= i) && (i < m_length));

在这里插入图片描述

让我们来解释一下:

(0 <= i) 意味着 i 必须大于或等于0,因为在大多数编程中,位置通常从0开始编号,所以不能小于0。

(i < m_length) 意味着 i 必须小于线性表的长度 m_length。这是因为线性表中的位置索引不能超过线性表的总元素个数。

这两个条件一起的意思是,我们需要确保 i 不仅大于等于0,还要小于线性表的长度,这样我们才能访问线性表中有效的位置。如果 i 不满足这两个条件,那么访问它可能导致越界错误,这是一种非常常见的编程错误,因此需要进行这样的合法性检查,以防止程序出现问题。

  • 将目标位置作为数组下标把参数设置进去
    因为我们这是原生数组,所以直接使用下标操作即可
m_array[i] = e;

总体代码如下:

bool set(int i, const T& e)
{
    bool ret = ((0 <= i) && (i < m_length));

    if (ret)
    {
        m_array[i] = e;
    }

    return ret;
}

在这里插入图片描述

3.3 insert函数

带2个参数的insert

顺序存储结构的元素插入操作
1.判断目标位置是否合法

bool ret = ((0 <= i) && (i <= m_length));  // 1

ret = ret && (m_length < capacity());  // 1

在这里插入图片描述

0 <= i 确保目标位置 i 不小于0,因为位置通常是从0开始计数的。
i <= m_length 保证目标位置 i 不超过线性表的当前长度,以免越界。
m_length < capacity() 确保线性表的长度没有超过它的容量,以防止插入元素导致溢出。
这两个步骤合在一起,就是在确认插入的位置 i 是一个有效、合法的位置,不会导致数组越界或者超过线性表的容量限制。

为什么是i <= m_length呢
拿出你的手,是不是一般有5个手指,那现在有一支笔要放到你手指中间有几种方式?
有6种,所以需要i<=m_length

2.将目标位置之后的所有元素后移一个位置

for(int p=m_length-1; p>=i; p--)   // n, 0
{
    m_array[p + 1] = m_array[p];
}

在这里插入图片描述

循环从线性表的最后一个元素开始,一直到目标位置 i。
m_array[p + 1] = m_array[p] 将每个元素向后移动一个位置,给插入的元素腾出空间。
这个步骤是为了给新元素腾出位置,确保插入操作不会覆盖掉原有的元素。通过将目标位置之后的元素都向后移动一个位置,为新元素腾出了插入的空间。这是线性表顺序存储结构中插入操作的一部分,确保插入后的线性表仍然是有序、没有元素遗漏的。

3.将新元素插入目标位置
4…线性表长度加 1

总体代码如下:

bool insert(int i, const T& e)
{
    bool ret = ((0 <= i) && (i <= m_length));
    ret = ret && (m_length < capacity());

    if (ret)
    {
        for (int p = m_length - 1; p >= i; p--)
        {
            m_array[p + 1] = m_array[p];
        }

        m_array[i] = e;
        m_length++;
    }

    return ret;
}

在这里插入图片描述

带一个参数的insert

那么这个函数设计出来干什么的呢?
其实就是方便我们尾插入的。

所以我们怎么实现他呢?
我们直接调用我们实现的2个参数的insert,且在位置的参数写上m_length,其意义就为尾插入

总体代码如下:

bool insert(const T& e) 
{
    insert(m_length, e);
}

在这里插入图片描述

3.4 remove函数

顺序存储结构的元素删除操作
1,判断目标位置是否合法

bool ret = ((0 <= i) && (i < m_length));

在这里插入图片描述

0 <= i 确保目标位置 i 不小于0,因为位置通常是从0开始计数的。
i < m_length 保证目标位置 i 不超过线性表的当前长度,以确保删除的位置是有效的。
这个步骤用于检查删除操作是否在合法的范围内进行,以防止删除不存在的元素或者越界。

2,将目标位置后的所有元素前移一个位置

for(int p=i; p<m_length-1; p++)   // n - 1
 {
     m_array[p] = m_array[p+1];
 }

在这里插入图片描述

循环从目标位置 i 开始,一直到线性表的倒数第二个元素(m_length-1位置)。
m_array[p] = m_array[p+1] 将每个元素向前移动一个位置,覆盖掉目标位置上的元素。
这个步骤是为了删除操作。通过将目标位置之后的元素都向前移动一个位置,可以覆盖掉要删除的元素,相当于将它从线性表中删除掉。删除后,线性表中的元素仍然是连续的,没有间隙,而且长度减少了一个元素。

3,线性表长度减 1

总体代码如下:

bool remove(int i)
{
    bool ret = ((0 <= i) && (i < m_length));

    if (ret)
    {
        for (int p = i + 1; p < m_length-1; p++)
        {
            m_array[p] = m_array[p + 1];
        }

        m_length--;
    }

    return ret;
}

在这里插入图片描述

3.5 clear函数

那么我们的clear操作是干什么的呢?
就是清空所有元素。

那么怎么实现呢?
可能有人知道,调用remove函数一个一个去删除。
那么这样的时间就会消耗更多。

我们不妨回到insert函数里面看看

在insert的实现中有这样一行代码,我们可以找到insert判断的条件的小技巧,直接把m_length设置0
那么我们是不是就变成初始状态了,所以就可以这样实现

bool ret = ((0 <= i) && (i <= m_length));

总体代码如下:

void clear()
{
    m_length = 0;
}

在这里插入图片描述

3.6 下标运算符重载函数

无const版本

我们的实现非常简单,和我们的get函数类似,只不过是我们需要在参数范围错误的时候抛出相应的异常即可

T& operator[](int i)
{
    if ((0 <= i) && (i < m_length))
    {
        return m_array[i];
    }
    else
        THROW_EXCEPTION(IndexOutOfBoundsException, "Parameter i is invalid...");
}

在这里插入图片描述

const版本

其实我们的const和非const是一样的代码,我们完全可以直接复制过来,但是为了代码的复用性我们可以这样写:

T operator[](int i)const
{
    return const_cast<SeqList<T>&>(*this)[i];
}

在这里插入图片描述

我们使用了const_cast去除本类的const属性,使他变成了一个普通的对象,然后我们使用[]运算,就会调用我们没有const的下标重载运算符了

3.7 length函数

int length() const
{
    return m_length;
}

在这里插入图片描述


总结

顺序存储结构是一种在计算机编程中常见的数据结构,它允许我们有效地存储和操作一系列数据元素。通过使用数组或向量等数据结构,我们可以实现顺序存储结构的基本操作,如插入、删除、查找和遍历。这种存储方式在许多应用中都非常有用,例如列表、栈、队列等。了解顺序存储结构的抽象实现有助于我们更好地理解和应用数据结构的概念。希望本文能够帮助您更好地理解顺序存储结构的基本原理和应用。

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

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

相关文章

97 只出现一次的数字

只出现一次的数字 题解1 异或的应用&#xff08;判断出现次数是奇偶&#xff09; 给你一个 非空 整数数组 nums &#xff0c;除了某个元素只出现一次以外&#xff0c;其余每个元素均出现两次。找出那个只出现了一次的元素。 你必须设计并实现线性时间复杂度的算法来解决此问题…

PCIe寄存器之二

关键字索引&#xff1a; CAP_PM 对应&#xff1a;Capabilities: [c0] Power Management CAP_MSI 对应&#xff1a;Capabilities: [c8] MSI CAP_MSIX对应&#xff1a;Capabilities: [e0] MSI-X CAP_EXP 对应&#xff1a;Capabilities: [70] Express (v2) Endpoint 以上 [] 内的…

开设自己的网站系类03安装数据库(centos版)

编者买了一个服务器打算自己构建一个网站&#xff0c;用于记录生活。网站大概算是一个个人博客吧。记录创建过程的一些步骤。 前面已经讲过配置服务器的程序运行环境 网站运行还需要数据库&#xff0c;本篇文章则是安装数据库的内容。 卸载mariadb 查看是否有安装 mariadb&…

【Python】数据分析案例:世界杯数据可视化

文章目录 前期数据准备导入数据 分析&#xff1a;世界杯中各队赢得的比赛数分析&#xff1a;先打或后打的比赛获胜次数分析&#xff1a;世界杯中的抛硬币决策分析&#xff1a;2022年T20世界杯的最高得分者分析&#xff1a;世界杯比赛最佳球员奖分析&#xff1a;最适合先击球或追…

【C语言基础】近期所学到的函数以及关键字(函数memset、scanf、关键字staric、 inline、volatile)

&#x1f4e2;&#xff1a;如果你也对机器人、人工智能感兴趣&#xff0c;看来我们志同道合✨ &#x1f4e2;&#xff1a;不妨浏览一下我的博客主页【https://blog.csdn.net/weixin_51244852】 &#x1f4e2;&#xff1a;文章若有幸对你有帮助&#xff0c;可点赞 &#x1f44d;…

python核心编程速记【笔记迁移】

笔记速记 1.python非常注重缩进&#xff0c;这是它的显著特征之一。 2.import相当于头文件声明模块。 3.利用type函数 type(a)可以查看当前变量类型。 isinstance可以比较两个数据类型并返回一个布尔值。 4.这里面的可直接使用and和or作为一个函数 5.python的算法比较贴合…

如何改善食品饮料包装生产企业的OEE?

食品饮料这类商品在我们的日常生活中十分常见&#xff0c;它们存在于各类商店、超市或路边的小店里。而食品饮料的包装是吸引人们购买该产品的一个重要因素。为了在这个市场中脱颖而出并提高盈利能力&#xff0c;企业需要关注设备的综合效率&#xff0c;即OEE&#xff08;Overa…

SpringBoot测试类启动web环境-上篇

1.坐标修改 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency> 2.测试类测试 说明&#xff1a;SpringBootTest()中的webEnvironment值的说明&#xff1b; 2.1不启…

vue项目js原生属性IntersectionObserver实现图片懒加载

vue项目js原生属性IntersectionObserver实现图片懒加载 IntersectionObserver 使用js原生属性IntersectionObserver实现观察img元素是否处于游览器视口中 懒加载原理&#xff1a;给img设置一个默认url图片&#xff0c;观察图片处于视口内以后&#xff0c;动态改变img的url为自己…

机组 CPU

控制器&#xff1a;协调并控制计算机各部件执行程序的指令序列&#xff0c;其基本功能是取指令、分析指令和执行指令 功能 CPU 必须具有控制程序的顺序执行&#xff08;指令控制&#xff09;、产生完成每条指令所需的控制命令&#xff08;操作控制&#xff09;、对各种操作加…

原神游戏干货分享:探索璃月的宝箱秘密,提高游戏资源获取效率!

《原神》是一款备受玩家喜爱的开放世界冒险游戏&#xff0c;而在游戏中获取资源是提升角色实力的重要途径。在这篇实用干货分享中&#xff0c;我们将介绍一些探索璃月地区的宝箱秘密&#xff0c;帮助你提高游戏资源获取的效率。 首先&#xff0c;璃月地区的宝箱分为普通宝箱和精…

Leetcode刷题详解—— 找出所有子集的异或总和再求和

1. 题目链接&#xff1a;1863. 找出所有子集的异或总和再求和 2. 题目描述&#xff1a; 一个数组的 异或总和 定义为数组中所有元素按位 XOR 的结果&#xff1b;如果数组为 空 &#xff0c;则异或总和为 0 。 例如&#xff0c;数组 [2,5,6] 的 异或总和 为 2 XOR 5 XOR 6 1 。…

基于QT使用OpenGL,加载obj模型,进行鼠标交互

目录 功能分析&#xff08;需求分析&#xff09;技术点分析OpenGL立即渲染模式可编程渲染管线模式 QOpenGLWidget派生类 glwidget逻辑glwidget.hglwidget.cpp 鼠标交互功能obj格式介绍 效果bunnyCayman_GT 功能分析&#xff08;需求分析&#xff09; 基于QT平台&#xff0c;使…

【被面试官吊打系列】啥,你没说面试要考智力题呀 (上) ?

你好&#xff0c;我是安然无虞。 文章目录 1. 二进制问题分金条问题毒药问题 2. 先手必胜问题轮流拿石子抢30的必胜策略Nim游戏 3. 水桶问题5L和6L的水桶怎么量出3L的水&#xff1f;3L和5L的水桶怎么量出4L的水&#xff1f;一个装了10L水的桶&#xff0c;一个7L的空桶还有一个…

2.【自动驾驶与机器人中的SLAM技术】左乘模型推导ESKF

目录 1. 证明题 证明&#xff1a;若某个高斯随机变量为零均值&#xff0c;协方差为对角线矩阵且大小相同&#xff08;各向同性&#xff09;&#xff0c;那么在乘任意旋转矩阵以后&#xff0c;其均值仍为零&#xff0c;且协方差不变&#xff1b; 2. 代码实现运动方程将F矩阵…

FreeRTOS_内存管理

目录 1. 内存管理简介 2. 内存碎片 3. heap_1 内存分配方法 3.1 分配方法简介 4. heap_2 内存分配方法 4.1 分配方法简介 4.2 内存块详解 5. heap_4 内存分配方法 6. FreeRTOS 内存管理实验 6.1 实验程序 内存管理是一个系统基本组成部分&#xff0c;FreeRTOS 中大量…

计算机考研408有多难?25考研经验贴,开个好头很有必要

前言 大家好&#xff0c;我是陈橘又青&#xff0c;相信关注我的各位小伙伴们中&#xff0c;大多都是在计算机专业的大学生吧&#xff01; 每天都有许多人在后台私信我&#xff0c;问我要不要考研&#xff0c;我想说这个东西是因人而异的&#xff0c;像我本人就选择了就业&…

istio 学习笔记

参考&#xff1a;istio简介和基础组件原理&#xff08;服务网格Service Mesh&#xff09;-CSDN博客 Istio 微服务框架 服务治理。 Istio的关键功能: HTTP/1.1&#xff0c;HTTP/2&#xff0c;gRPC和TCP流量的自动区域感知负载平衡和故障切换。 通过丰富的路由规则&#xf…

96 前缀树Trie

前缀树 题解1 STL题解2 参考官方 Trie&#xff08;发音类似 “try”&#xff09;或者说 前缀树 是一种树形数据结构&#xff0c;用于高效地存储和检索字符串数据集中的键。这一数据结构有相当多的应用情景&#xff0c;例如自动补完和拼写检查。 请你实现 Trie 类&#xff1a; …

php冒泡算法实现倒序和正序排列

冒泡排序是一种简单的排序算法&#xff0c;其主要思想是比较相邻的两个元素&#xff0c;根据需要交换位置&#xff0c;将较大&#xff08;或较小&#xff09;的元素逐渐冒泡到数组的一端&#xff0c;从而实现排序。 1、从小到大排序 function bubbleSort($arr) {$len count(…