LeetCode题练习与总结:基本计算器 Ⅱ--227

news/2024/9/18 21:15:34/ 标签: 算法, 数据结构, LeetCode, Java, , 数学, 字符串

一、题目描述

给你一个字符串表达式 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 整数

二、解题思路

这个问题可以使用(stack)来解决。我们使用两个,一个用于存储操作数,另一个用于存储操作符。遍历字符串s,根据遇到的字符类型(数字、操作符或空格)进行相应的处理。

  1. 当遇到数字时,将其转换为整数并存储在操作数中。
  2. 当遇到操作符时,需要考虑操作符的优先级。如果当前操作符的优先级高于顶操作符的优先级,则直接将当前操作符入。否则,需要先计算顶的操作符,并将结果重新入,然后比较当前操作符与新的顶操作符的优先级。
  3. 当遇到空格时,忽略它。
  4. 对于乘法和除法,因为它们的优先级高于加法和减法,所以一旦遇到乘号或除号,需要立即计算结果,并将结果入
  5. 对于加法和减法,可以直接将操作数入,因为它们可以在最后一起计算。

以下是具体的步骤:

  • 初始化两个:一个用于存储操作数(nums),另一个用于存储操作符(ops)。
  • 遍历字符串s,对于每个字符c
    • 如果c是空格,则忽略。
    • 如果c是数字,则解析出完整的数字并压入nums
    • 如果c是操作符(+-*/):
      • 如果ops为空或者顶的操作符是(,则直接将c压入ops
      • 如果c*/,则从nums中弹出两个操作数进行计算,并将结果压回nums
      • 如果c+-,并且ops顶不是(,则从nums中弹出两个操作数进行计算,并将结果压回nums,然后将c压入ops
  • 遍历完成后,如果ops中还有操作符,则继续从nums中弹出操作数进行计算,直到ops为空。
  • 最后,nums中的唯一元素即为表达式的结果。

三、具体代码

import java.util.Stack;public class Solution {public int calculate(String s) {Stack<Integer> nums = new Stack<>();Stack<Character> ops = new Stack<>();for (int i = 0; i < s.length(); i++) {char c = s.charAt(i);if (c == ' ') continue;if (Character.isDigit(c)) {int num = 0;while (i < s.length() && Character.isDigit(s.charAt(i))) {num = num * 10 + (s.charAt(i) - '0');i++;}i--; // since the for loop also increments inums.push(num);} else if (c == '(') {ops.push(c);} else if (c == ')') {while (ops.peek() != '(') {nums.push(applyOp(ops.pop(), nums.pop(), nums.pop()));}ops.pop();} else if (c == '+' || c == '-' || c == '*' || c == '/') {while (!ops.isEmpty() && hasPrecedence(c, ops.peek())) {nums.push(applyOp(ops.pop(), nums.pop(), nums.pop()));}ops.push(c);}}while (!ops.isEmpty()) {nums.push(applyOp(ops.pop(), nums.pop(), nums.pop()));}return nums.pop();}public boolean hasPrecedence(char op1, char op2) {if (op2 == '(' || op2 == ')') return false;if ((op1 == '*' || op1 == '/') && (op2 == '+' || op2 == '-')) return false;return true;}public int applyOp(char op, int b, int a) {switch (op) {case '+': return a + b;case '-': return a - b;case '*': return a * b;case '/': return a / b;}return 0;}
}

请注意,代码中的applyOp方法假设中的操作数顺序是正确的,并且对于减法和除法,ab的顺序很重要(a是被操作数,b是操作数)。代码也处理了括号的情况,确保在遇到右括号时,所有括号内的表达式都会被计算。

四、时间复杂度和空间复杂度

1. 时间复杂度
  • 遍历字符串s:假设字符串s的长度为n,则遍历字符串的时间复杂度为O(n)
  • 解析数字:在遍历过程中,每次遇到数字,需要解析出完整的数字。在最坏的情况下,数字可能连续出现,这意味着对于每个数字字符,我们需要进行一次操作。因此,解析数字的总操作次数也是O(n)
  • 应用操作符:在遍历字符串的过程中,每次遇到操作符,我们可能需要从中弹出元素进行计算,然后将结果压回中。在最坏的情况下,可能需要对每个操作符都进行这样的操作。由于每个操作符最多被处理一次,因此这一步的时间复杂度也是O(n)

综上所述,总的时间复杂度为O(n),其中n字符串s的长度。

2. 空间复杂度
  • 操作数nums:在最坏的情况下,所有数字都直接入,没有进行任何计算,因此的大小可能达到O(n)
  • 操作符ops:在最坏的情况下,所有操作符都直接入,没有进行任何计算,因此的大小可能达到O(n)

综上所述,总的空间复杂度为O(n),其中n字符串s的长度。

这里的O(n)表示算法的复杂度与输入字符串的长度成线性关系。在实际情况中,由于数字不会连续占据整个字符串,操作符和操作数的大小通常会小于n,但最坏情况下的复杂度分析仍然是O(n)

五、总结知识点

  • (Stack)的使用

  • 字符和字符串操作

    • 使用char类型来处理单个字符。
    • 使用String类的charAt方法来获取字符串中指定位置的字符。
    • 使用Character.isDigit方法来判断一个字符是否是数字。
  • 数字解析

    • 通过循环和字符减去’0’的方式,将字符串中的连续数字字符转换为整数。
  • 运算符优先级

    • 在处理算术表达式时,需要根据运算符的优先级来决定计算的顺序。
    • 使用hasPrecedence方法来判断当前运算符与顶运算符的优先级关系。
  • 条件逻辑

    • 使用if-else语句来处理不同类型的字符(空格、数字、运算符、括号)。
    • 使用while循环来处理连续的数字字符,以及应用运算符。
  • 异常处理

    • 虽然在给出的代码中没有显示异常处理,但在实际应用中,应当考虑除法操作可能导致的ArithmeticException(如除以零)。
  • 递归下降

    • 在处理括号时,实际上隐式地使用了递归下降解析的原理,通过递归地将括号内的表达式视为一个子表达式来处理。
  • 运算符应用

    • 使用applyOp方法来根据运算符执行相应的算术运算。
  • 算法逻辑

    • 理解和实现一个基本的算术表达式求值算法,这涉及到算法设计的基本概念。
  • 类型转换

    • 在将字符转换为整数时,涉及到从charint的类型转换。

以上就是解决这个问题的详细步骤,希望能够为各位提供启发和帮助。


http://www.ppmy.cn/news/1525797.html

相关文章

深度挖掘| 如何高效实现Cloudera 安装之基础环境搭建

Cloudera Manager是CDH市场领先的管理平台。它以其强大的数据管理和分析能力&#xff0c;帮助企业能够轻松驾驭海量数据&#xff0c;实现数据的实时分析与洞察。 作为业界第一的端到端 Apache Hadoop 的管理应用&#xff0c;Cloudera Manager对CDH的每个部件都提供了细粒度的可…

软件测试工程师面试整理-编程与自动化

在软件测试领域,编程与自动化是提升测试效率、覆盖率和可靠性的关键因素。掌握编程技术和自动化测试框架,能够帮助测试人员有效地执行大量重复性测试任务,并迅速反馈软件的质量状况。以下是编程与自动化在测试中的主要应用及相关技术介绍: 1. 编程语言与自动化 ● 常用编程…

k8s dashboard token 生成/获取

创建示例用户 在本指南中&#xff0c;我们将了解如何使用 Kubernetes 的服务帐户机制创建新用户、授予该用户管理员权限并使用与该用户绑定的承载令牌登录仪表板。 对于以下每个和的代码片段ServiceAccount&#xff0c;ClusterRoleBinding您都应该将它们复制到新的清单文件(如)…

Java应用压测工具JMeter

目录 1、下载JMeter 2、配置环境变量 3、配置语音 4、使用 1、下载JMeter Apache JMeter - Apache JMeter™ 千万别下载这个&#xff0c;会报错、 千万别下载这个&#xff0c;会报错、 千万别下载这个&#xff0c;会报错 下载这个&#xff0c;失败多下载几次 2、配置环…

http的请求方式

HTTP协议支持多种请求方法&#xff0c;每种方法都有其特定的用途和场景。以下是HTTP中最常用的几种请求方法及其用途&#xff1a; ‌‌GET‌&#xff1a;用于请求服务器发送资源&#xff0c;通常用于请求数据。GET请求常用于获取信息&#xff0c;如查询操作&#xff0c;其参数附…

明天考教资之作文素材

草木不经霜雪&#xff0c;则生意不顾&#xff1b;吾人不经忧患&#xff0c;则德惠不成。人生的大海波澜壮阔&#xff0c;纵然狂风骤雨卷起千堆雪&#xff0c;海也只当吹皱了春水&#xff0c;永远吹不走的&#xff0c;是那颗永恒的直面磨难的心。 追风赶月莫停留&#xff0c;平芜…

ISP住宅网络的特点是什么

在互联网服务领域&#xff0c;ISP&#xff08;互联网服务提供商&#xff09;住宅网络是连接家庭用户与互联网世界的重要桥梁。随着技术的发展和用户需求的增长&#xff0c;ISP住宅网络已经成为现代生活中不可或缺的一部分。本文将探讨ISP住宅网络的特点、它所带来的优势以及在不…

【CSS|第1期】网页设计的演变:从表格布局到Grid布局

日期&#xff1a;2024年9月9日 作者&#xff1a;Commas 签名&#xff1a;(ง •_•)ง 积跬步以致千里,积小流以成江海…… 注释&#xff1a;如果您觉在这里插入代码片得有所帮助&#xff0c;帮忙点个赞&#xff0c;也可以关注我&#xff0c;我们一起成长&#xff1b;如果有不对…

分治算法之归并排序详细解读(附带Java代码解读)

分治算法中的归并排序&#xff08;Merge Sort&#xff09; **归并排序&#xff08;Merge Sort&#xff09;是一种基于分治法&#xff08;Divide and Conquer&#xff09;**的排序算法。其核心思想是将数组递归地分成若干个子数组&#xff0c;分别对每个子数组进行排序&#xf…

【计算机网络 - 基础问题】每日 3 题(四)

✍个人博客&#xff1a;Pandaconda-CSDN博客 &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/fYaBd &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 C 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;收藏&…

25届计算机专业选题推荐-基于python的线上拍卖会管理系统【python-爬虫-大数据定制】

&#x1f496;&#x1f525;作者主页&#xff1a;毕设木哥 精彩专栏推荐订阅&#xff1a;在 下方专栏&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; 实战项目 文章目录 实战项目 一、基于python的线上拍卖会管理…

老古董Lisp实用主义入门教程(8):挠痒痒先生建网站记

是时候来个真正的应用 几位奇形怪状, 百无聊赖的先生, 用Common Lisp 搞东搞西一阵子, 总觉得没有干什么正经事. 一般而言, 学习编程语言总是应该先搞点计算, 让CPU燥起来. 但是Lisp搞计算总感觉有点不太对劲, 虽然颠倒先生已经尝试把数学公式改成中序以增强动力, 但是不行. 隔…

鸿蒙 ArkUI组件一

ArkUI组件 布局 布局指用特定的组件或者属性来管理用户页面所放置UI组件的大小和位置。在实际的开发过程中&#xff0c;需要遵守以下流程保证整体的布局效果&#xff1a; 确定页面的布局结构。分析页面中的元素构成。选用适合的布局容器组件或属性控制页面中各个元素的位置和大…

MySQL 数据库:原理、应用与发展

摘要&#xff1a;本文深入探讨了 MySQL 数据库相关内容。首先介绍了 MySQL 作为开源关系型数据库管理系统的显著特点&#xff0c;包括易用性、跨平台性、高性能、可扩展性、开源免费以及数据安全性等方面。接着详细阐述了其安装与配置过程&#xff0c;涵盖在不同操作系统上的安…

[晕事]今天做了件晕事44 wireshark 首选项IPv4:Reassemble Fragented IPv4 datagrams

不知不觉&#xff0c;已经来到了晕事系列的第四十四个晕事。今天办的晕事和Wireshark查看网络包相关。说&#xff0c;在Wireshark的编辑-首选项协议里的IPv4协议&#xff0c;有一个参数设置是&#xff1a;Reassemble Fragented IPv4 datagrams。 这个参数的含义是指定Wireshar…

Linux通配符*、man 、cp、mv、echo、cat、more、less、head、tail、等指令、管道 | 、指令的本质 等的介绍

文章目录 前言一、Linux通配符*二、man 指令三、 cp 指令四、mv指令五、 echo 指令六、cat 指令七、more 指令八、 less 指令九、 head 指令十、 tail指令十一、 管道 |十二、指令的本质总结 前言 Linux通配符*、man 、cp、mv、echo、cat、more、less、head、tail、等指令、管…

vue2,3生命周期

Vue.js 的生命周期在 Vue 2 和 Vue 3 中有所不同&#xff0c;但基本的概念是相似的。Vue 的生命周期是指 Vue 实例从创建到销毁的整个过程&#xff0c;这个过程中 Vue 实例会触发一系列的事件&#xff0c;我们称之为生命周期钩子&#xff08;Lifecycle Hooks&#xff09;。开发…

在Milvus中创建集合并在集合中插入数据,然后attu管理工具可以查看

日志打印出来的是这个&#xff0c;现在attu为什么看不到插入的数据信息&#xff0c;集合信息已经可以看到&#xff0c;为什么看不到数据呢/home/anaconda3/envs/bi-txt-sql/bin/python -X pycache_prefix/home/.cache/JetBrains/PyCharm2023.2/cpython-cache /home/tools/pycha…

前端——标签二(超链接)

标签二 超链接标签&#xff1a;a 超链接&#xff0c;实现页面间的跳转和数据传输 a标签的属性 href&#xff1a;跳转路径&#xff08;url&#xff09;必须具备&#xff0c;表示点击后会跳转到哪个页面 target&#xff1a;页面打开方式。默认是 _self 如果是 _blank则用新的…

CSDN玩法攻略(维护中)

以下均为测试过的条件 隐形条件和官方描写可能不准确更新不及时 勋章 签到勋章(已下架) 勤写标兵 每周三篇原创等级1 max10 创作能手 lv1 每周1-3 lv2 每周4-6 lv3 每周7-8 lv4 每周>9 持续创作 授予每个自然月内发布4篇或4篇以上原创或翻译IT博文的用户 五一创作勋章 每…