227. 基本计算器 II Python

文章目录


一、题目描述

给你一个字符串表达式 s ,请你实现一个基本计算器来计算并返回它的值。

整数除法仅保留整数部分。

你可以假设给定的表达式总是有效的。所有中间结果将在 [-2^31, 2^31 - 1]的范围内。

注意 :不允许使用任何将字符串作为数学表达式计算的内置函数,比如 eval()

示例 1

输入:s = "3+2*2"
输出:7

示例 2

输入:s = " 3/2 "
输出:1

示例 3

输入:s = " 3+5 / 2 "
输出:5

提示:
1 <= s.length <= 3 * 10^5
s 由整数和算符 ('+', '-', '*', '/') 组成,中间由一些空格隔开
s 表示一个 有效表达式
表达式中的所有整数都是非负整数,且在范围 [0, 2^31 - 1] 内
题目数据保证答案是一个 32-bit 整数

二、代码

代码如下:

c 复制代码
class Solution:
    def calculate(self, s: str) -> int:
        def is_operator(char):
            return char in "+-*/^"

        def infix_to_postfix(expression):
            def precedence(operator):
                precedence_dict = {'+': 1, '-': 1, '*': 2, '/': 2, '^': 3}
                return precedence_dict.get(operator, 0)

            def infix_to_postfix_internal(expression_tokens):
                output = []
                operator_stack = []

                for token in expression_tokens:
                    if token.isnumeric():  # 如果是数字,直接添加到输出
                        output.append(token)
                    elif token == '(':  # 如果是左括号,压入栈
                        operator_stack.append(token)
                    elif token == ')':  # 如果是右括号,将栈顶的操作符弹出并添加到输出,直到遇到左括号
                        while operator_stack and operator_stack[-1] != '(':
                            output.append(operator_stack.pop())
                        if operator_stack and operator_stack[-1] == '(':
                            operator_stack.pop()
                    elif is_operator(token):  # 如果是操作符,处理操作符的优先级
                        while (operator_stack and
                            operator_stack[-1] != '(' and
                            precedence(token) <= precedence(operator_stack[-1])):
                            output.append(operator_stack.pop())
                        operator_stack.append(token)

                while operator_stack:  # 将栈中剩余的操作符全部添加到输出
                    output.append(operator_stack.pop())

                return ' '.join(output)

            # 将输入的表达式字符串按空格分割成标记列表
            expression_tokens = expression.split()
            # 调用内部函数进行转换
            postfix_expression = infix_to_postfix_internal(expression_tokens)
            return postfix_expression

        def add_spaces_to_infix(expression):
            operators = "+-*/^"
            result = []

            for char in expression:
                if char in operators or char in "()":
                    result.append(f" {char} ")
                else:
                    result.append(char)

            return ''.join(result)

        infix_expression = s.replace(" ", "")
        print(infix_expression)
        spaced_infix_expression = add_spaces_to_infix(infix_expression)
        print(spaced_infix_expression)
        postfix_expression = infix_to_postfix(spaced_infix_expression)
        print("中缀表达式:", spaced_infix_expression)
        print("后缀表达式:", postfix_expression)

        stack = []

        for token in postfix_expression.split():
            if token.isnumeric():
                stack.append(int(token))
            elif is_operator(token):
                operand2 = stack.pop()
                operand1 = stack.pop()
                if token == '+':
                    result = operand1 + operand2
                elif token == '-':
                    result = operand1 - operand2
                elif token == '*':
                    result = operand1 * operand2
                elif token == '/':
                    if operand2 == 0:
                        raise ValueError("Division by zero")
                    result = int(operand1 / operand2)
                stack.append(result)

        print(int(stack[0]))
        return int(stack[0])

三、解题思路

本题本质是要求通过字符串来计算表达式,且不能直接使用eval方法对字符串直接进行转化计算。本题解题思路为:将字符串(前缀表达式)转化为后缀表达式,然后通过计算后缀表达式得到结果。

① 因为涉及到字符可能会出现2位数及其以上的情况,如果之间转化为后缀表达式,则会导致数字连接到一块,所以需要对数字和运算符进行分隔,将表达式转化为如下格式:"12+4/5" => "12 + 4 / 5"

② 转化为后缀表达式,当遇到2位数及以上的数字时,需要将其看做是一个数,用空格分隔不同数。例如:

中缀表达式: 12 + 4 / 5

后缀表达式: 12 4 5 / +

不同的数之间用空格分开

③ 计算后缀表达式,找第一个运算符,向左找最近的2个数进行计算,重复这一过程,最后得到一个值返回即可。

相关推荐
唐叔在学习几秒前
【唐叔学算法】第21天:超越比较-计数排序、桶排序与基数排序的Java实践及性能剖析
数据结构·算法·排序算法
程序员_三木4 分钟前
Three.js入门-Raycaster鼠标拾取详解与应用
开发语言·javascript·计算机外设·webgl·three.js
是小崔啊14 分钟前
开源轮子 - EasyExcel01(核心api)
java·开发语言·开源·excel·阿里巴巴
ALISHENGYA20 分钟前
全国青少年信息学奥林匹克竞赛(信奥赛)备考实战之分支结构(switch语句)
数据结构·算法
tianmu_sama20 分钟前
[Effective C++]条款38-39 复合和private继承
开发语言·c++
chengooooooo21 分钟前
代码随想录训练营第二十七天| 贪心理论基础 455.分发饼干 376. 摆动序列 53. 最大子序和
算法·leetcode·职场和发展
黄公子学安全23 分钟前
Java的基础概念(一)
java·开发语言·python
liwulin050624 分钟前
【JAVA】Tesseract-OCR截图屏幕指定区域识别0.4.2
java·开发语言·ocr
jackiendsc28 分钟前
Java的垃圾回收机制介绍、工作原理、算法及分析调优
java·开发语言·算法
Oneforlove_twoforjob32 分钟前
【Java基础面试题027】Java的StringBuilder是怎么实现的?
java·开发语言