[数据结构]栈的实现与应用

文章目录

  • 一、引言
  • 二、栈的基本概念
    • 1、栈是什么
    • 2、栈的实现方式对比
    • 3、函数栈帧
  • 三、栈的实现
    • 1、结构体定义
    • 2、初始化
    • 3、销毁
    • 4、显示
    • 5、数据操作
  • 四、分析栈
    • 1、优点
    • 2、缺点
  • 五、总结
    • 1、练习题
    • 2、源代码

一、引言

栈,作为一种基础且重要的数据结构,在计算机科学领域中有着广泛的应用。它不仅为函数调用提供了必要的支持,还在算法设计和问题解决中发挥着关键作用。本文将对栈的基本概念、实现方式以及函数栈帧进行详细的介绍和分析。


二、栈的基本概念

1、栈是什么

栈是一种特殊的线性数据结构,其只允许在固定的一端进行插入和删除操作。插入和删除的这一端被称为栈顶,而另一端则被称为栈底。栈遵循后进先出的原则,即最后插入的元素会最先被删除。这种特性使得栈在算法设计和问题解决中具有独特的优势。

请添加图片描述

2、栈的实现方式对比

  • 数组实现:将数组的尾部充当栈顶,数据的插入与删除操作变得非常高效。同时数组的结构,使得其CPU高速缓存的命中率较高。唯一的缺陷就是,扩容或缩容会有一定的性能开销。
  • 链表实现:采用单链表结构,将链表头部设为栈顶,数据的插入与删除操作同样能高效完成。但链表需要额外的存储空间来保存指针信息,且由于链表的结构,CPU高速缓存的命中率相对较低。此外,链表实现的复杂度相较于数组实现也更高一些。

3、函数栈帧

在函数调用过程中,系统会为每个函数分配一个栈帧。栈帧中存储了函数的局部变量、参数以及返回地址等信息。当函数执行完毕后,其栈帧会被销毁,从而释放占用的栈空间。函数栈帧的分配和销毁过程体现了栈的后进先出原则。

函数栈帧的存在使得函数调用具有嵌套性,即一个函数可以调用另一个函数,而被调用的函数又可以继续调用其他函数。这种嵌套调用关系通过栈来维护,确保了函数调用的正确性和稳定性。


三、栈的实现

1、结构体定义

首先,定义了栈的数据结构。栈使用动态数组来存储元素,同时记录了栈顶索引和栈的容量。

typedef int DataType;
typedef struct Stack {
	DataType* array;
    int top;
	int capacity;
}S;

2、初始化

接下来,实现了栈的初始化函数。该函数为栈分配内存,并初始化栈顶索引和容量。

void Init(S* ps, int capacity)
{
    assert(ps != NULL && capacity >= 0);

    DataType* pos = (DataType*)malloc(sizeof(DataType) * capacity);
    if (pos == NULL)
    {
        fprintf(stderr, "内存分配失败");
        exit(EXIT_FAILURE);
    }

    ps->array = pos;
    ps->capacity = capacity;
    ps->top = -1;
}

3、销毁

实现了栈的销毁函数。该函数释放栈所占用的内存,并将栈的指针、容量和栈顶索引重置为初始状态。

void Destroy(S* ps)
{
    if (ps == NULL)
        return;

    free(ps->array);
    ps->array = NULL;
    ps->capacity = 0;
    ps->top = -1;
}

4、显示

实现了栈的显示函数。该函数遍历栈中的元素,并调用用户定义的打印函数来打印每个元素。

void Print(S* ps, void (*Prin)(DataType))
{
    assert(ps != NULL);

    for (int i = ps->top; i >= 0; i--)
    {
        Prin(ps->array[i]);
    }
    printf("\n");
}

5、数据操作

void Push(S* ps, DataType data)
{
    assert(ps != NULL);

    if (ps->top == ps->capacity - 1)
    {
        int capacity = ps->capacity == 0 ? 2 : ps->capacity * 2;
        DataType* pos = (DataType*)realloc(ps->array, sizeof(DataType) * capacity * 2);
        if (pos == NULL)
        {
            fprintf(stderr, "内存分配失败");
            exit(EXIT_FAILURE);
        }

        ps->array = pos;
        ps->capacity = capacity;
    }

    ps->array[++(ps->top)] = data;
}

void Pop(S* ps)
{
    assert(ps != NULL && ps->top != -1);

    if (ps->capacity > 64 && ps->top < ps->capacity / 3)
    {
        int capacity = ps->capacity / 3;

        DataType* pos = (DataType*)realloc(ps->array, sizeof(DataType) * capacity);
        if (pos == NULL)
        {
            fprintf(stderr, "内存分配失败");
            exit(EXIT_FAILURE);
        }

        ps->array = pos;
        ps->capacity = capacity;
    }

    ps->top--;
}

DataType Top(S* ps)
{
    assert(ps != NULL);

    return ps->array[ps->top];
}

四、分析栈

1、优点

  • 操作简便:栈提供了简洁的接口,如push(入栈)和pop(出栈),使得元素的操作非常直观和方便。
  • 高效性:对于大多数栈实现,push和pop操作的时间复杂度均为O(1),保证了高效的元素访问速度。
  • 应用场景广泛:栈在多种算法和数据结构中都有重要应用,如深度优先搜索(DFS)、表达式求值、括号匹配等。

2、缺点

  • 访问限制:栈只允许在栈顶进行元素的插入和删除操作,无法直接访问栈内的其他元素,限制了其灵活性。

五、总结

1、练习题

  • 有效的括号
  • 最小栈
  • 用栈操作构建数组
  • 堆盘子

2、源代码

对于栈的源代码我已经开源在GItee:传送门。


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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【数据结构】滑动窗口算法详解:高效解决子串问题

滑动窗口&#xff08;Sliding Window&#xff09;是一种常用于处理数组或字符串中子序列问题的算法技巧。它通过维护一个窗口来限制待处理的数据范围&#xff0c;从而高效地解决问题&#xff0c;避免重复计算。它的时间复杂度通常为 O(N)&#xff0c;相较于暴力破解&#xff08…

Go 项目如何集成类似mybatisPlus插件呢?GORM走起!!

导读&#xff1a; 在 Go 项目中&#xff0c;虽然没有像 MyBatis Plus 这样特定的 ORM 插件&#xff0c;但可以使用功能相似的 Go ORM 框架&#xff0c;比如 GORM&#xff0c;它支持链式查询、自动迁移、预加载等功能&#xff0c;与 MyBatis Plus 有相似之处。通过一些插件或扩…

常用API

Object类&#xff1a; instanceof&#xff1a;java中的关键字&#xff0c;判断左边的对象是否是右面类的实例。 它的作用是判断其左边对象是否为其右边类的实例&#xff0c;返回boolean类型的数据。 getClass()&#xff1a;得到调用者的数据类型&#xff1b; 进行对象内容比较…

016_基于python+django网络爬虫及数据分析可视化系统2024_kyz52ks2

目录 系统展示 开发背景 代码实现 项目案例 获取源码 博主介绍&#xff1a;CodeMentor毕业设计领航者、全网关注者30W群落&#xff0c;InfoQ特邀专栏作家、技术博客领航者、InfoQ新星培育计划导师、Web开发领域杰出贡献者&#xff0c;博客领航之星、开发者头条/腾讯云/AW…

盘点现代浏览器的各种神奇能力,功能令人惊讶

盘点现代浏览器的各种神奇能力&#xff0c;功能令人惊讶&#x1f62e; 浏览器的进化 一个运行在浏览器里面的操作系统。一个炫酷的量子纠缠网页。内嵌在浏览器里面的AI大模型。 随着web技术的迅猛发展&#xff0c;现代浏览器已经不仅仅是一个浏览网页的工具了。它的功能早已进…

【判断推理】逻辑论证之归因论证

2.1 归因论证概述 归因&#xff1a;指人们对 他人或自己行为的原因的推论过程。具体而言&#xff0c;就是观察者对他人的行为过程或自己的行为过程所进行的因果解释和推论。&#xff08;通俗而言&#xff0c;归因就是对已经发生的事实&#xff0c;在众多可能的原因中找出一个原…

Cesium 实战 - 自定义纹理材质 - 立体墙(旋转材质)

Cesium 实战 - 自定义纹理材质 - 立体墙(旋转材质) 核心代码完整代码在线示例Cesium 给实体对象(Entity)提供了很多实用的样式,基本满足普通项目需求; 但是作为 WebGL 引擎,肯定不够丰富,尤其是动态效果样式。 对于实体对象(Entity),可以通过自定义材质,实现各种…

CLion和Qt 联合开发环境配置教程(Windows和Linux版)

需要安装的工具CLion 和Qt CLion下载链接 :https://www.jetbrains.com.cn/clion/ 这个软件属于直接默认安装就行&#xff0c;很简单&#xff0c;不多做介绍了 Qt:https://mirrors.tuna.tsinghua.edu.cn/qt/official_releases/online_installers/ window 直接点exe Linux 先c…

【2024软著申请】软著申请到发放全流程(附带教程+工具+撰写建议)

目录 总时间线材料准备1、计算机软件著作权登记申请表4页2、身份证明文件3、软件鉴别材料4、文档鉴别材料 唠叨两句 总时间线 时间关键节点20240811电子材料提交进入待受理阶段20240826受理阶段审查通过&#xff0c;进入审查中20240930发放完成 材料准备 版权登记链接(https…

用柔性神经k-Opt学习搜索路径问题的可行和不可行区域(未完,先看前驱文章L2S)

文章目录 Abstract1 IntroductionAbstract 介绍了一种名为 Neural k-Opt(NeuOpt)的新型学习搜索(L2S)求解器,用于解决路径问题。它学习执行基于定制的动作分解方法和定制的循环双流(Recurrent Dual-Stream)解码器的灵活 k-opt 交换。 作为一项开创性的工作,我们绕过了…

华山论剑之Rust的Trait

华山论剑&#xff0c;群雄荟萃&#xff0c;各显神通。武林中人&#xff0c;各有所长&#xff0c;或剑法飘逸&#xff0c;或掌法刚猛&#xff0c;或轻功绝顶。这就好比Rust中的trait&#xff0c;它定义了一种武功套路&#xff0c;而不同的门派、不同的人&#xff0c;可以将这套武…

All-reduce,AIl-to-all

目录 跨中心架构下的大模型并行训练 优化All-reduce通信效率 优化AIl-to-all通信效率 跨中心架构下的大模型并行训练 优化All-reduce通信效率 All-reduce是一种在分布式计算中广泛使用的通信操作,用于将多个节点的数据聚合成一个全局结果,并将该结果分发回所有节点。优化…

sv标准研读第十五章-进程间同步与通信

书接上回&#xff1a; sv标准研读第十四章-clocking block 第15章 进程间的同步和通信 15.1 概览 -semaphores -mailboxes -named events 15.2 综述 简单的进程间通信可以通过named events来实现&#xff0c;有event trigger和event control过程&#xff0c;分别需要依…

Elasticsearch基本使用及介绍

Elasticsearch 1. 关于各种数据库的使用 关于MySQL&#xff1a;是关系型数据库&#xff0c;能清楚的表示数据之间的关系&#xff0c;并且&#xff0c;是基于磁盘存储的&#xff0c;可以使用相对较低的成本存储大量的数据 关于Redis&#xff1a;是基于K-V结构的在内存中读写数…

2011年国赛高教杯数学建模B题交巡警服务平台的设置与调度解题全过程文档及程序

2011年国赛高教杯数学建模 B题 交巡警服务平台的设置与调度 有困难找警察”&#xff0c;是家喻户晓的一句流行语。警察肩负着刑事执法、治安管理、交通管理、服务群众四大职能。为了更有效地贯彻实施这些职能&#xff0c;需要在市区的一些交通要道和重要部位设置交巡警服务平台…

【jQuery】jQuery 处理 Ajax 以及解决跨域问题的方式

文章目录 HTTP原生创建 AjaxjQuery 处理 Ajax$.ajax()$().load()$.get()$.post() 跨域CORSJSONPiframeweb sockets HTTP 超文本传输协议&#xff08;HTTP&#xff0c;HyperText Transfer Protocol)是互联网上应用最为广泛的一种网络协议。设计 HTTP 最初的目的是为了提供一种发…

QT中中文显示乱码问题

在VS2013中用QT开发GUI应用程序&#xff0c;Qt中显示中文乱码 一&#xff1a; //解决QT中中文显示乱码问题 #pragma execution_character_set("utf-8") 二&#xff1a;在main函数中添加以下代码&#xff1a; #include <QTextCodec>void main() {QTextCod…

javaweb-mybatis之动态sql

(1).if标签 编写好方法之后&#xff0c;选中方法名&#xff0c;alt回车&#xff0c;选第一个generate statement快捷生成xml里的标签 (2).foreach标签 用于批量删除 (3)sql和include标签

别再犯这些Java并发编程的常见错误!你中了几个?

你好&#xff0c;我是忆~遂愿&#xff0c;全网2w粉丝&#xff0c;《遂愿盈创》社群主理人。 副业启航① | 遂愿盈创&#xff08;对副业感兴趣免费可入&#xff0c;多种赚钱实战项目等你来&#xff0c;一起探寻副业快速变现的途径&#xff1b;以及对接互联网大厂商务合作&#x…

YOLO11改进 | 主干网络 | 将backbone替换为Swin-Transformer结构【论文必备】

秋招面试专栏推荐 &#xff1a;深度学习算法工程师面试问题总结【百面算法工程师】——点击即可跳转 &#x1f4a1;&#x1f4a1;&#x1f4a1;本专栏所有程序均经过测试&#xff0c;可成功执行&#x1f4a1;&#x1f4a1;&#x1f4a1; 本文给大家带来的教程是将YOLO11的backb…