昨天去某大厂面试,居然让我做四则运算,还好我够机灵。

简介: 面试官:请您说说怎么计算四则运算?比如1 + 2 * ( 3 + 4 ) - 5。我:先算括号里再算括号外,先乘除后加减,最后等于10。面试官懵了一下,说:可能我说明白,我想问的是用计算机怎么计算?我尬尴的笑了笑,马上说到:对于计算机来说,单纯的两个数的加减乘除很容易,但是如果乘除在加减的后面却要先运算,再加上几个括号,就变得更加复杂了。为了使计算机更容易理解,前人已为我们引入了一种新的四则运算的表示法。

面试官:请您说说怎么计算四则运算?比如1 + 2 * ( 3 + 4 ) - 5

我:先算括号里再算括号外,先乘除后加减,最后等于10。

面试官懵了一下,说:可能我说明白,我想问的是用计算机怎么计算?

我尬尴的笑了笑,马上说到:对于计算机来说,单纯的两个数的加减乘除很容易,但是如果乘除在加减的后面却要先运算,再加上几个括号,就变得更加复杂了。

为了使计算机更容易理解,前人已为我们引入了一种新的四则运算的表示法。

逆波兰式

有一位波兰数学家名字叫:扬·武卡谢维奇(Jan Łukasiewicz),就是这位:

Jan Łukasiewicz

他在1920年引入一种新的数学表达式形式:逆波兰式(Reverse Polish notation,RPN,或逆波兰记法)。在逆波兰式中,不需要括号来标识操作符的优先级,所有操作符都放在操作数的后面,因此也被称为后缀表达法。我们在小学时经常用的是中缀表达法,也就是操作符在操作数之间,比如:1 + 2 * ( 3 + 4 ) - 5,对应的逆波兰式就是:1 2 3 4 + * + 5 -

逆波兰式的计算

根据逆波兰式计算结果,需要借助栈(Stack)这种数据结构。栈(Stack)是限定仅在表尾进行插入或删除操作的线性表,遵循后进先出(Last In First Out,LIFO)的原则。

具体步骤如下:

  1. 从左到右扫描表达式,如果当前字符为数字,则入栈。
  2. 如果当前字符为运算符,则将栈顶两个元素出栈,作相应运算,结果再入栈。
  3. 最后当表达式扫描完后,栈里的就是计算结果了。

我们以1 2 3 4 + * + 5 -为例:

输入 操作 堆栈 注释
1 入栈 1
2 入栈 1,2
3 入栈 1,2,3
4 入栈 1,2,3,4
+ 加法运算 1,2,7 3,4出栈,将结果7入栈
* 乘法运算 1,14 2,7出栈,将结果14入栈
+ 加法运算 15 1,14出栈,将结果15入栈
5 入栈 15,5
- 减法运算 10 15,5出栈,将结果10入栈

最终的计算结果为:10。

逆波兰式的转换

计算机可以根据逆波兰式计算出结果,那么问题来了!我们常用的中缀表达法怎么转换成逆波兰式呢?也是需要借助栈这种数据结构,具体步骤如下:

  1. 从左到右扫描中缀表达式,如果当前字符为数字就直接输出。
  2. 如果当前字符为运算符,则判断其与栈顶运算符的优先级。
  3. 如果是右括号或者优先级低于栈顶运算符,则栈内运算符依次出栈并输出,然后当前运算符入栈。
  4. 最后当中缀表达式扫描完后,输出的就是逆波兰式了。

我们以1 + 2 * ( 3 + 4 ) - 5为例:

输入 操作 堆栈 输出
1 输出 1
+ 入栈 + 1
2 输出 + 1 2
* 入栈 +,* 1 2
( 入栈 +,*,( 1 2
3 输出 +,*,( 1 2 3
+ 入栈 +,*,(,+ 1 2 3
4 输出 +,*,(,+ 1 2 3 4
) 依次出栈至( +,* 1 2 3 4 +
- 依次出栈,然后-入栈 - 1 2 3 4 + * +
5 输出 - 1 2 3 4 + * + 5
依次出栈 1 2 3 4 + * + 5 -

最终的逆波兰式为:1 2 3 4 + * + 5 -

看着面试官满意的笑容,我擦了擦额头汗,还好我够机灵。

感谢帅哥靓女的 关注收藏点赞
相关文章
|
算法 C++ Python
【每日算法Day 66】经典面试题:不用四则运算如何做加法?
【每日算法Day 66】经典面试题:不用四则运算如何做加法?
135 0
|
机器学习/深度学习 自然语言处理 算法
|
5月前
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
2月前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
|
2月前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
2月前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
79 4
|
3月前
|
算法 Java 数据中心
探讨面试常见问题雪花算法、时钟回拨问题,java中优雅的实现方式
【10月更文挑战第2天】在大数据量系统中,分布式ID生成是一个关键问题。为了保证在分布式环境下生成的ID唯一、有序且高效,业界提出了多种解决方案,其中雪花算法(Snowflake Algorithm)是一种广泛应用的分布式ID生成算法。本文将详细介绍雪花算法的原理、实现及其处理时钟回拨问题的方法,并提供Java代码示例。
103 2
|
3月前
|
JSON 安全 前端开发
第二次面试总结 - 宏汉科技 - Java后端开发
本文是作者对宏汉科技Java后端开发岗位的第二次面试总结,面试结果不理想,主要原因是Java基础知识掌握不牢固,文章详细列出了面试中被问到的技术问题及答案,包括字符串相关函数、抽象类与接口的区别、Java创建线程池的方式、回调函数、函数式接口、反射以及Java中的集合等。
43 0
|
5月前
|
XML 存储 JSON
【IO面试题 六】、 除了Java自带的序列化之外,你还了解哪些序列化工具?
除了Java自带的序列化,常见的序列化工具还包括JSON(如jackson、gson、fastjson)、Protobuf、Thrift和Avro,各具特点,适用于不同的应用场景和性能需求。
|
5月前
|
Java
【Java基础面试三十七】、说一说Java的异常机制
这篇文章介绍了Java异常机制的三个主要方面:异常处理(使用try、catch、finally语句)、抛出异常(使用throw和throws关键字)、以及异常跟踪栈(异常传播和程序终止时的栈信息输出)。

热门文章

最新文章