【深入解析:数据结构栈的魅力与应用】

本章重点

  • 栈的概念及结构

  • 栈的实现方式

  • 数组实现栈接口

  • 栈面试题目

  • 概念选择题

一、栈的概念及结构

栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端 称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。

压栈:栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶

出栈:栈的删除操作叫做出栈。出数据也在栈顶

  • 栈顶Top:线性表允许插入和删除的那一端。
  • 栈底Bottom:固定的,不允许进行插入和删除的另一端。
  • 空栈:不含任何元素的空表。

二、栈的实现方式

        由于栈只能在栈顶操作,所以栈的实现一般可以使用数组或者链表实现,相对而言数组的结构实现更优一些。因为数组在尾上插入数据的代价比较小。单链表的尾插需要遍历结点,这里使用带头双向循环链表也可,但是空间销毁大。单链表也可以实现,我们就需要让栈顶是头结点,栈底是尾结点,这样入栈就是头插,效率高。

1.顺序堆栈


        根据前边的分析可知,顺序堆栈和顺序表的数据成员(元素)是相同的,不同之处是,顺序堆栈的入栈和出栈操作只能对当前栈顶元素进行。
        顺序堆栈的存储结构如图3-2所示。其中,a0,a1,a2,表示顺序堆栈要存储的元素序列,stack 表示顺序堆栈存放元素的数组,capacity 表示顺序堆栈数组stack的内存单元容量(表示目前允许存储的元素最大个数),top表示顺序堆栈数组stack的当前栈顶位置。

2.链式堆栈

        堆栈有两端,插入元素和删除元素的一端为栈顶,另一端为栈底。对链式堆栈来说,显然,若把靠近头指针的一端定义为栈顶,则插入元素和删除元素时不需要遍历整个链,其时间复杂度为0(1);若把远离头指针的一端定义为栈顶,则每次插入元素和删除元素时都需要遍历整个链,其时间复杂度为0(n)。因此,链式堆栈都设计成把靠近头指针的一端定义为栈顶。链式堆栈的头结点对操作实现的影响不大,因此可有可无。依次向带头结点链式堆栈输入a0,a1,a2,…….an-1后,带头结点链式堆栈的结构示意图如图3-3所示。

三、数组实现栈接口

// 下面是定长的静态栈的结构,实际中一般不实用,所以我们主要实现下面的支持动态增长的栈
typedef int STDataType;
#define N 10
typedef struct Stack
{STDataType a[N];int top; // 栈顶
}Stack;
// 支持动态增长的栈
typedef int STDataType;
typedef struct Stack
{STDataType* a;int top; // 栈顶int capacity; // 容量
}Stack;// 初始化栈
void StackInit(Stack* ps);
// 入栈
void StackPush(Stack* ps, STDataType data);
// 出栈
void StackPop(Stack* ps);
// 获取栈顶元素
STDataType StackTop(Stack* ps);
// 获取栈中有效元素个数
int StackSize(Stack* ps);
// 检测栈是否为空,如果为空返回0,如果不为空返回非零结果
int StackEmpty(Stack* ps);
// 销毁栈
void StackDestroy(Stack* ps);

1.初始化栈:void StackInit(Stack* ps)

初始化这里我们先不设置容量,后续入栈再申请空间即可,这里需要注意top的值,我们设置的值是0,它是表示非空栈中的栈顶指针始终在栈顶元素的下一个位置上。

// 初始化栈
void StackInit(Stack* ps)
{assert(ps);ps->a = NULL;ps->top = 0;//表示栈顶元素的下一个位置ps->capacity = 0;
}

2.入栈:void StackPush(Stack* ps, STDataType data)

满栈:当我们使一个元素入栈的之前,我们往往需要判断一下栈是否为满栈,防止发生上溢的情况。因为我们定义了一个capacity来表示当前已经分配的存储空间,所以我们可以用ps->top == ps->capacity 来判断当前使用的栈空间是否满了。所以当ps->top == ps->capacity时表示已经满了。

满栈我们要首先追加存储空间,然后才能将元素入栈。realloc()函数可以申请空间,如果realloc第一个参数为空,那么realloc的功能就类似于malloc,这也是为什么我们前面初始化没有开辟空间的原因,写在这里能省大量的代码。

// 入栈
void StackPush(Stack* ps, STDataType data)
{assert(ps);if (ps->top == ps->capacity){int newCapacity = (ps->capacity == 0) ? 4 : ps->capacity * 2;//如果ps->a == NULL,功能相当与mallocSTDataType* tmp = (STDataType*)realloc(ps->a, sizeof(STDataType) * newCapacity);if (tmp == NULL){perror("realloc fail");exit(-1);}ps->a = tmp;ps->capacity = newCapacity;}ps->a[ps->top] = data;ps->top++;
}

3.出栈:void StackPop(Stack* ps)

出栈时我们首先要判断栈是否为空栈,由于是顺序栈,我们只需要判断ps->top > 0即可判断当前栈的情况。

// 出栈
void StackPop(Stack* ps)
{assert(ps);//保证不为空assert(ps->top > 0);ps->top--;
}

4.获取栈顶元素:STDataType StackTop(Stack* ps)

取栈顶元素同样需要判断是否为空栈,空栈也不能取出任何数据,所以这里强制断言,一旦为空栈直接程序报错

// 获取栈顶元素
STDataType StackTop(Stack* ps)
{assert(ps);//保证不为空assert(ps->top > 0);return ps->a[ps->top - 1];
}

5.获取栈中有效元素个数:int StackSize(Stack* ps)

由于top是非空栈中的栈顶指针始终在栈顶元素的下一个位置上,所以top就是当前栈中有效元素个数

// 获取栈中有效元素个数
int StackSize(Stack* ps)
{assert(ps);return ps->top;
}

6.检测栈是否为空,如果为空返回0,如果不为空返回非零结果:int StackEmpty(Stack* ps)

// 检测栈是否为空,如果为空返回0,如果不为空返回非零结果
int StackEmpty(Stack* ps)
{if (ps->top != 0)return ps->top;elsereturn 0;
}

7.销毁栈:void StackDestroy(Stack* ps)

销毁栈只需将申请的空间释放,再把设置的大小和容量设施为0即可。

// 销毁栈
void StackDestroy(Stack* ps)
{assert(ps);free(ps);ps->a = NULL;ps->top = 0;ps->capacity = 0;
}

四、栈面试题目

括号匹配问题。OJ链接

        当开始接触题目时,我们会不禁想到如果计算出左括号的数量,和右括号的数量,如果每种括号左右数量相同,会不会就是有效的括号了呢?

        事实上不是的,假如输入是 [ { ] },每种括号的左右数量分别相等,但不是有效的括号。这是因为结果还与括号的位置有关。

        仔细分析我们发现,对于有效的括号,它的部分子表达式仍然是有效的括号,比如 { ( )[ ) ] } 是一个有效的括号,( )[ { } ] 是有效的括号,[ ( ) ] 也是有效的括号。并且当我们每次删除一个最小的括号对时,我们会逐渐将括号删除完。比如下面的例子。

这个思考的过程其实就是栈的实现过程。因此我们考虑使用栈,当遇到匹配的最小括号对时,我们将这对括号从栈中删除(即出栈),如果最后栈为空,那么它是有效的括号,反之不是。

bool isValid(char* s) 
{Stack st;StackInit(&st);char topVal;while (*s){switch (*s){case'(':case'[':case'{':StackPush(&st, *s);break;case')':case']':case'}'://数量不匹配if (!StackEmpty(&st))//栈为空{StackDestroy(&st);//防止内存泄露return false;}topVal = StackTop(&st);StackPop(&st);//顺序不匹配if (*s == ')' && topVal != '(' || *s == ']' && topVal != '['|| *s == '}' && topVal != '}'){StackDestroy(&st);//防止内存泄露return false;}break;			}++s;}//栈不为空,此时只有右括号,false,说明数量不匹配int ret = StackEmpty(&st);StackDestroy(&st);if (ret == 0)return false;elsereturn true;
}

五、概念选择题

1.一个栈的初始状态为空。现将元素1、2、3、4、5、A、B、C、D、E依次入栈,然后再依次出栈,则元素出栈的顺序是( )。

  • A 、12345ABCDE
  • B 、EDCBA54321
  • C 、ABCDE12345
  • D 、54321EDCBA

元素出栈的顺序遵循后进先出(Last-In-First-Out,LIFO)原则,因此选择B

2.若进栈序列为 1,2,3,4 ,进栈过程中可以出栈,则下列不可能的一个出栈序列是()

  • A 1,4,3,2
  • B 2,3,4,1
  • C 3,1,4,2
  • D 3,4,2,1

A:1入栈后出栈,2,3,4入栈,4出栈,3出栈,2出栈,可能。

B:1,2入栈,2出栈,3入栈,3出栈,4入栈,4出栈,1出栈,可能。

C:1,2,3入栈,3出栈,接下来应该是2出栈或者4入栈,不可能1出栈,不可能。

D:1,2,3入栈,3出栈,4入栈,4出栈,2出栈,1出栈,可能。

本章结束啦!!!

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

常用系统命令

重定向 cat aa.txt > bbb.txt 将输出定向到bbb.txt cat aaa.txt >> bbb.txt 输出并追加查看进程 ps ps -ef 显示所有进程 例⼦:ps -ef | grep mysql |:管道符 kill pid 结束进程, 如 kill 3732;根据进程名结束进程可以先…

易服客工作室:Uncode主题 - 创意和WooCommerce WordPress主题

Uncode主题是一款像素完美的创意 WordPress 主题,适用于任何类型的网站(作品集、代理机构、自由职业者、博客),也是适用于商店(电子商务、在线商店、企业)的顶级 WooCommerce 主题。Uncode 的设计非常注重细…

【hive】简单介绍hive的几种join

文章目录 前言1. Common Join2. Map Join介绍:使用方法:限制: 3. Bucket Map Join介绍:好处:使用条件:使用方法: 4. Sort Merge Bucket Map Join介绍:如何使用: 5. Skew …

“开发和运维”只是一个开始,最终目标是构建高质量的软件工程

随着技术的飞速发展,软件行业不断寻求改进和创新的方法来提供更高质量的产品。在这方面,DevOps已经展现出了巨大的潜力。通过打破开发和运维之间的壁垒,DevOps将持续集成、持续交付和自动化流程引入到软件开发中,使团队能够更快地…

android cocoscreator 检测模拟器还是真机

转载至 一行代码帮你检测Android模拟器 具体原理看原博主文章,这里只讲cocoscreator3.6的安卓工程怎么使用 1.新建一个com.lahm.library包,和com.cocos.game同目录,如图示 那四个文件的代码如下: EmulatorCheckUtil类&#…

JVM基础了解

JVM 是java虚拟机。 作用:运行并管理java源码文件锁生成的Class文件;在不同的操作系统上安装不同的JVM,从而实现了跨平台的保证。一般在安装完JDK或者JRE之后,其中就已经内置了JVM,只需要将Class文件交给JVM即可 写好的…

ModaHub魔搭社区:AI Agent在操作系统场景下的AgentBench基准测试

近日,来自清华大学、俄亥俄州立大学和加州大学伯克利分校的研究者设计了一个测试工具——AgentBench,用于评估LLM在多维度开放式生成环境中的推理能力和决策能力。研究者对25个LLM进行了全面评估,包括基于API的商业模型和开源模型。 他们发现,顶级商业LLM在复杂环境中表现出…

CSS 选择器

前言 基础选择器 以下是几种常见的基础选择器。 标签选择器&#xff1a;通过HTML标签名称选择元素。 例如&#xff1a; p {color: red; } 上述样式规则将选择所有<p>标签 &#xff0c;并将其文字颜色设置为红色。 类选择器&#xff1a;通过类名选择元素。使用类选择…

Android Retrofit原理浅析

官方地址:Retrofit 原理:Retrofit 本质上是代理了OKhttp,使用代理模式,Type-Safe 类型安全 编译器把类型检查出 避免类型错误, enqueue 异步 切换线程 execute 同步 不切换线程 enqueue:Call接口定义的抽象方法 Retrofit.Create() 方法首先验证接口validateServiceInterf…

iOS设计规范是什么?都有哪些具体规范

iOS设计规范是苹果为移动设备操作系统iOS制定的设计指南。iOS设计规范的制定保证了苹果应用在外观和操作上的一致性和可用性&#xff0c;从而提高了苹果界面设计的用户体验和应用程序的成功性。本文将从七个方面全面分析iOS设计规范。 1.iOS设计规范完整版分享 由「即时设计」…

批量爬虫采集完成任务

批量爬虫采集是现代数据获取的重要手段&#xff0c;然而如何高效完成这项任务却是让许多程序员头疼的问题。本文将分享一些实际操作价值高的方法&#xff0c;帮助你提高批量爬虫采集的效率和专业度。 目标明确&#xff0c;任务合理划分&#xff1a; 在开始批量爬虫采集前&…

剪枝基础与实战(3): 模型剪枝和稀疏化训练流程

Model Pruning 相关论文:Learning Efficient Convolutional Networks through Network Slimming (ICCV 2017) 考虑一个问题,深度学习模型里面的卷积层出来之后的特征有非常多,这里面会不会存在一些没有价值的特征及其相关的连接?又如何去判断一个特征及其连接是否有价值? …