波兰表达式

基本概念

从栈的操作特性来看,栈是一种操作受限的线性表,只允许在一端插入和删除数据。

事实上,从功能上来说,数组和链表完全可以替代栈的使用。但是,从某种角度来说,数组和链表暴露太多的操作接口,实现起来比较复杂,也很容易出错。

因此,当数据集合只涉及在一端插入和删除数据,并且满足先进后出、后进先出的特性,可以首选“栈”这种数据结构

操作

栈是允许在同一端进行插入和删除操作的特殊线性表。以下是栈的一些概念:

  • 允许进行插入和删除操作的一端称为栈顶,栈顶会根据插入、删除操作进行浮动
  • 不允许进行插入和删除操作的一端称为栈底,栈底是固定不变的
  • 栈中元素个数为零时称为空栈
  • 往栈中插入元素称作进栈
  • 删除栈顶的元素称作出栈

栈的实现

实际上,栈既可以用数组实现,也可以用链表实现。使用数组实现的栈被称为顺序栈,使用链表实现的栈被称为链式栈。

顺序栈

顺序栈结构

使用数组作为栈存储数据的物理存储结构,需要先申请一个大小为 n 的数组。

将数组尾部作为栈顶,使用一个变量表示数组存储的元素个数,这样入栈、出栈的操作都能达到O(1) O(1)\的时间复杂度。

基于数组实现的栈存在一个限制,即数组在声明之后是固定大小的,当栈满的时候,则无法再次往数组中添加数据。这样就涉及到对数组进行动态扩容,当数组空间不够的时候,需要重新申请一块更大的内存,将原来数组中数据统统拷贝过去。

事实上,支持动态扩容的顺序栈,在实际开发中并不常见。一般使用到顺序栈这种数据结构的情况,都是一些规模比较小的数据,有时候知道数据的最大规模时也可以避免爆栈。

 

链式栈

链式栈结构

对于数据规模比较大的情况,使用链表实现栈则会更加方便。这时候,不需要提前申请内存空间,而是创建一个哨兵结点指向头结点。

链式栈一般会使用头插法创建链表。将头结点作为栈顶,每次入栈都当作链表在头结点插入元素,每次出栈都当作链表删除头结点,这些操作都只需要处理好哨兵结点的指向即可。链式栈一般都能达到O(1) O(1)\的时间复杂度。

基于链表实现的栈不需要像数组一样要动态扩容,链表是天生支持动态扩容的。

应用场景

栈作为一个比较基础的数据结构,应用场景还是蛮多的。

函数调用栈

函数调用栈是一个非常经典的应用场景。

操作系统给每个线程分配了一块独立的内存空间,这块内存被组织成“栈”这种结构,用来存储函数调用时的临时变量。每进入一个函数,就会将临时变量作为一个栈帧入栈,当被调用函数执行完成,返回到上层函数之后,再将这个函数对应的栈帧出栈。

直到函数栈为空,则表示最外层的函数已经执行完毕。

 

通常的四则运算表达式写成 (1 + 2) x (3 + 4)加减乘除等运算符写在中间,因此也被称作“中缀表达式”;逆波兰表达式的写法是 1 2 + 3 4 + x运算符被写在后面,因而也被称作“后缀表达式”;除此之外,还有波兰表达式,其写法是 x + 1 2 + 3 4,也被称作“前缀表达式”。

如果将表达式画出一棵语法树,就能以树的概念来直观理解前缀、中缀、后缀的含义,前缀表达式对应树的前序遍历,中缀表达式对应树的中序遍历,后缀表达式对应树的后序遍历,如下图所示:

首先,需要先将中缀表达式转换成逆波兰表达式,涉及到操作数栈和运算符栈:

  1. 从左向右遍历中缀表达式;
  2. 若读取的是操作数,将操作数存入到操作数栈中;
  3. 若读取的是运算符,还需要根据运算符类型进一步判断:
    1. 如果是 ( 运算符,则直接存入运算符栈中;
    2. 如果是 ) 运算符,则输出运算符栈中的运算符到操作数栈,直到遇到 ( 运算符,在这里放弃 () 运算符;
    3. 如果是非括号运算符,还需要与运算符栈栈顶的运算符比较优先级:
      1. 如果运算符栈栈顶的运算符是括号,则直接存入运算符栈中;
      2. 如果此运算符比运算符栈栈顶的运算符优先级更高时,则直接存入运算符栈中;
      3. 如果此运算符比运算符栈栈顶的运算符优先级更低或相等时,则输出栈顶运算符到操作数栈,然后将当前运算符压入操作符栈;
  4. 当表达式读取完后,将运算符栈依次取出到操作数栈中,直到运算符栈为空。

然后对得到的逆波兰表达式计算,得到表达式的结果:

  1. 从左向右遍历逆波兰表达式;
  2. 若读取的是操作数,将操作数存入栈中;
  3. 若读取的是运算符,则对栈顶的两个操作数执行该运算,然后将运算结果存入栈中;
  4. 重复上述的步骤 1~3,最终栈中即为结果值。



作者:程序员翔仔
链接:https://www.zhihu.com/question/41103160/answer/2446411216
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值