表达式求值问题(java)

简介: 表达式求值问题(java)

题目:


给定一个表达式,其中运算符仅包含 +,-,*,/(加 减 乘 整除),可能包含括号,请你求出表达式的最终值。


注意:


数据保证给定的表达式合法。


题目保证符号 - 只作为减号出现,不会作为负号出现,例如,-1+2,(2+2)*(-(1+1)+2) 之类表达式均不会出现。


题目保证表达式中所有数字均为正整数。


题目保证表达式在中间计算过程以及结果中,均不超过 231−1231−1。


题目中的整除是指向 00 取整,也就是说对于大于 00 的结果向下取整,例如 5/3=15/3=1,对于小于 00 的结果向上取整,例如 5/(1−4)=−15/(1−4)=−1。


C++和Java中的整除默认是向零取整;Python中的整除//默认向下取整,因此Python的eval()函数中的整除也是向下取整,在本题中不能直接使用。


样例:


输入: (2+2)*(1+1)输出: 8

思路:


此题考虑用栈,为中序遍历,有两个关键点


双栈,一个数字栈存取数字,一个运算符栈存取所有运算符


那么如何入栈呢?


数字直接入栈


括号优先级最高,遇到左括号直接入栈


遇到右括号代表括号结束,则进行运算操作,进行几次运算呢? 直到运算符栈的栈顶是左括号,具体来说就是数字栈出栈两个数,运算符栈出栈一个运算符,将运算结果入运算符栈。


如果是运算符


如果运算符栈为空,直接入栈


如果该运算符优先级大于运算符栈顶优先级,那么入栈


如果优先级小于等于运算符栈栈顶元素,则进行运算操作。直到运算符栈为空或者优先级大于运算符栈的栈顶元素


这个方法的时间复杂度为O(n),整个字符串只需要扫描一遍。


如果对栈的操作不熟悉,可以看看我的这篇文章


https://blog.csdn.net/m0_68055637/article/details/128593504?spm=1001.2014.3001.5502


代码:


import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.HashMap;
import java.util.Stack;
public class Main {
    static Stack<Integer> nums=new Stack<>();   //数字栈
    static Stack<Character> ops=new Stack<>();   //运算符栈
    public static void main(String[] args) throws IOException {
        BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
        HashMap<Character,Integer> map=new HashMap<>(); //存放运算符优先级
        map.put('+',1); map.put('-',1);
        map.put('*',2); map.put('/',2);
        char [] chars=br.readLine().toCharArray(); //接收表达式并转换成字符数组
        br.close();
        for (int i = 0; i < chars.length; i++) {
            char c=chars[i];
            if(Character.isDigit(c)){  //如果是数字类型的话
                int x=0,j=i;
                while (j< chars.length && Character.isDigit(chars[j])){
                    x=x*10+chars[j]-'0';
                    j++;
                }
                nums.push(x); //数字入栈
                i=j-1;
            }else if(c=='('){  //左括号直接入栈
              ops.push(c);
            }else if(c==')'){ //遇到右括号代表括号结束计算括号里面的
                while (ops.peek()!='(') {
                    eval();
                }
                ops.pop();  //左括号出栈
            }else {
                //待入栈运算符优先级低,那么先计算就行
                while(!ops.isEmpty() && ops.peek() != '(' && map.get(ops.peek()) >= map.get(c)){
                    eval();
                }
               ops.push(c);  //操作符入栈
            }
        }
        while (!ops.isEmpty()) eval(); //剩余的进行计算
        System.out.println(nums.peek());
    }
    private static void eval() {
        int b=nums.pop();
        int a=nums.pop();
        char c=ops.pop();
        if(c=='+') {
            nums.push(a+b);
        }else if(c=='-') {
            nums.push(a-b);
        }else if(c=='*'){
            nums.push(a*b);
        }else {
            nums.push(a/b);
        }
    }
}


易错点:


 //第21行
          while (j< chars.length && Character.isDigit(chars[j])){
               x=x*10+chars[j]-'0';
               j++;
          }
          nums.push(x); //数字入栈
           i=j-1;


这个地方注意j是下标,是从i开始,判断是否是数字的时候要加上chars[]数组

这里i=j-1,i不能等于j,假如如果这个数字是两位数,那么j=2,i如果等于j,if结束后i++,i=3,就跳过一个字符了


//第36行 
while(!ops.isEmpty() && ops.peek() != '(' && map.get(ops.peek()) >= map.get(c))


这个地方要加上判断栈顶不能为(


假设计算(1+3),先把(压栈到op进去,再把1压栈到num进去,接下来读取字符+,op栈里不为空为真(因为有(符号在栈里),判断栈顶元素不是左括号为假,假如不判断这个左括号的话,会继续执行while里的判断语句,把这个符号与栈顶元素的优先级比较,但是左括号没有优先级,会报错

————————————————

版权声明:本文为CSDN博主「小新要努力变强」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/m0_68055637/article/details/128646977

相关文章
|
3月前
|
深入解析java正则表达式
本文深入解析Java正则表达式的应用,从基础概念到实际开发技巧全面展开。正则表达式是一种强大的文本处理工具,广泛应用于格式验证、搜索替换等场景。Java通过`Pattern`和`Matcher`类支持正则表达式,`Pattern.compile()`方法将正则字符串编译为高效模式对象。文章详细介绍了核心类的功能、常用正则语法及实际案例(如邮箱和电话号码验证)。掌握这些内容,可显著提升文本处理能力,满足多种开发需求。
97 1
|
3月前
|
Java Lambda 表达式:以 Foo 接口为例深入解析
本文深入解析了 Java 8 中 Lambda 表达式的用法及其背后的函数式接口原理,以 `Foo` 接口为例,展示了如何通过简洁的 Lambda 表达式替代传统匿名类实现。文章从 Lambda 基本语法、函数式接口定义到实际应用层层递进,并探讨默认方法与静态方法的扩展性,最后总结常见误区与关键点,助你高效优化代码!
83 0
|
4月前
|
怎么理解Java中的lambda表达式
Lambda表达式是JDK8引入的新语法,用于简化匿名内部类的代码写法。其格式为`(参数列表) -&gt; { 方法体 }`,适用于函数式接口(仅含一个抽象方法的接口)。通过Lambda表达式,代码更简洁灵活,提升Java的表达能力。
|
4月前
|
《从头开始学java,一天一个知识点》之:运算符与表达式:算术、比较和逻辑运算
**你是否也经历过这些崩溃瞬间?** - 看了三天教程,连`i++`和`++i`的区别都说不清 - 面试时被追问&quot;`a==b`和`equals()`的区别&quot;,大脑突然空白 - 写出的代码总是莫名报NPE,却不知道问题出在哪个运算符 这个系列为你打造Java「速效救心丸」,每天1分钟,地铁通勤、午休间隙即可完成学习。直击高频考点和实际开发中的「坑位」,拒绝冗长概念,每篇都有可运行的代码示例。明日预告:《控制流程:if-else条件语句实战》。
75 6
Java中的Lambda表达式与Stream API的协同作用
在本文中,我们将探讨Java 8引入的Lambda表达式和Stream API如何改变我们处理集合和数组的方式。Lambda表达式提供了一种简洁的方法来表达代码块,而Stream API则允许我们对数据流进行高级操作,如过滤、映射和归约。通过结合使用这两种技术,我们可以以声明式的方式编写更简洁、更易于理解和维护的代码。本文将介绍Lambda表达式和Stream API的基本概念,并通过示例展示它们在实际项目中的应用。
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
深入理解Java中的Lambda表达式
在Java 8中引入的Lambda表达式,不仅简化了代码编写,还提升了代码可读性。本文将带你探索Lambda表达式背后的逻辑与原理,通过实例展示如何高效利用这一特性优化你的程序。
探索Java中的Lambda表达式
本文将深入探讨Java 8引入的Lambda表达式,这一特性极大地简化了代码编写,提高了程序的可读性。通过实例分析,我们将了解Lambda表达式的基本概念、使用场景以及如何优雅地重构传统代码。文章不仅适合初学者,也能帮助有经验的开发者加深对Lambda表达式的理解。
探索Java中的Lambda表达式
【10月更文挑战第37天】本文将带你深入理解Java的Lambda表达式,从基础语法到高级特性,通过实例讲解其在函数式编程中的应用。我们还将探讨Lambda表达式如何简化代码、提高开发效率,并讨论其在实际项目中的应用。
|
8月前
|
Java中的Lambda表达式与函数式编程####
【10月更文挑战第29天】 本文将深入探讨Java中Lambda表达式的实现及其在函数式编程中的应用。通过对比传统方法,我们将揭示Lambda如何简化代码、提高可读性和维护性。文章还将展示一些实际案例,帮助读者更好地理解和应用Lambda表达式。 ####
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问