Java实现简单的表达式计算器功能示例

 更新时间:2018年06月29日 12:11:30   作者:HiBoyljw  
这篇文章主要介绍了Java实现简单的表达式计算器功能,结合实例形式分析了Java针对输入表达式的符号分解与数值运算相关操作技巧,需要的朋友可以参考下

本文实例讲述了Java实现简单的表达式计算器功能。分享给大家供大家参考,具体如下:

读入一个只包含 +, -, *, / 的非负整数计算表达式,计算该表达式的值。

Input

测试输入包含若干测试用例,每个测试用例占一行,每行不超过200个字符,整数和运算符之间用一个空格分隔。没有非法表达式。当一行中只有0时输入结束,相应的结果不要输出。

Output

对每个测试用例输出1行,即该表达式的值,精确到小数点后2位。

Sample Input

1 + 2
4 + 2 * 5 - 7 / 11
0

Sample Output

3.00
13.36

这是一个简单的计算器。下面我自己讲一下原理吧。简单的来说就是弄两个栈一个用来存数值,一个用来存加减乘除的符号。你也可以用数组来做,不过JAVA自己自带栈的功能,用起来会方便很多。

值得注意的是:用来装数值的栈,最多存两个数值,这是为了计算。而装符号的则是装一个。当我们遇到' *' 和'/‘时,我们是先进行运算,再压栈。对了,在写运算函数时一定要注意,运算的顺序,我自己写的时候,因为大意,也是被坑了很久。

具体代码如下:

package Temp;
import java.util.Scanner;
import java.util.Stack;
public class P1237 {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String str = sc.nextLine();
    while (!str.equals("0")) {// 到0了就结束 用栈来做
    Stack<Double> num = new Stack<Double>();// 加数值
    Stack<Character> md = new Stack<Character>();// 加运算符“+”,“-”,“*”,“/"
    String str1[] = str.split(" ");// 用一个新的数组,以空格来划分
    for (int i = 0; i < str1.length; i++) {
      if (isNumber(str1[i])) {// 如果是数字就加栈
      double d = Double.parseDouble(str1[i]);
      if (num.size() <= 1) {// 最少能放俩个数字进去
        num.push(d);
      }
      } else {
      if (md.isEmpty()) {// 如果加字符的为空就加入进去
        md.push((str1[i]).charAt(0));
      } else {
        char md1 = md.peek();// 看栈的顶层是什么符号
        char md2 = str1[i].charAt(0);// i i的符号
        /*
         * 下面自己写个函数来根据返回值来判断,先运行什么符号
         */
        if (opration(md1, md2) <= 1) {
        double nextNum = num.pop();// 进行弹栈来运算//代表栈的顶端
        double lastNum = num.pop();// 代表栈的底端
        // 因为是小于等于一所以先把MD中的符号弹栈,再把后面的符号压栈,再把结果放入num中
        num.push(math(lastNum, nextNum, md.pop()));
        md.push(str1[i].charAt(0));
        /* 可以不要 */
        // } else if (opration(md1, md2) == 2// 判断下一个符号
        // && i + 2 < str1.length
        // && opration((str1[i + 2]).charAt(0), md2) != 1) {
        // double nextNum = Double.valueOf(str1[i + 1]);
        // double lastNum = num.pop();
        // num.push(math(nextNum, lastNum, md2));
        } else if (opration(md1, md2) == 2) {
        double nextNum = Double.valueOf(str1[++i]);// 获得下一个
        double lastNum = num.pop();
        num.push(math(lastNum, nextNum, md2));
        }
      }
      }
    }
    if (!md.empty()) {// 运算符栈中不空的话,数值栈中一定有俩个数值
      double nextNum = num.pop();
      double lastNum = num.pop();
      // System.out.printf("%.2f", math(lastNum, nextNum, md.pop()));
      System.out.println(String.format("%.2f",
        math(lastNum, nextNum, md.pop())));
    } else if (num.size() == 1) {
      // System.out.printf("%.2f\n", num.pop());
      System.out.println(String.format("%.2f", num.pop()));
    }
    str = sc.nextLine();// 获取下一个运算
    }
  }
  private static Double math(double num1, double num2, Character pop) {
    // 用swith case来进行匹配运算
    switch (pop) {
    case '+':
    return (num1 + num2);
    case '-':
    return num1 - num2;
    case '*':
    return num1 * num2;
    case '/':
    return num1 / num2;
    }
    return (double) 0;
  }
  private static int opration(char md1, char md2) {
    if (md1 == '*' || md1 == '/') {
    if (md2 == '*' || md2 == '/') {
      return 0;// 如果两个都是乘法或者除法那么运算顺序是一样的
    } else
      return 1;// 那先运行md1
    } else {
    if (md2 == '*' || md2 == '/') {
      return 2;// 先运行md2
    }
    }
    return 0;// 都是加加减法顺序一样
  }
  // 判断是否是数字
  private static boolean isNumber(String str) {
    char ch[] = str.toCharArray();
    for (int i = 0; i < str.length(); i++) {
    if (ch[i] < 48 || ch[i] > 57) {
      return false;
    }
    }
    return true;
  }
}

PS:这里再为大家推荐几款计算工具供大家参考:

在线数学表达式简单转换/计算工具:
http://tools.jb51.net/jisuanqi/exp_jisuanqi

在线一元函数(方程)求解计算工具:
http://tools.jb51.net/jisuanqi/equ_jisuanqi

科学计算器在线使用_高级计算器在线计算:
http://tools.jb51.net/jisuanqi/jsqkexue

在线计算器_标准计算器:
http://tools.jb51.net/jisuanqi/jsq

更多关于java算法相关内容感兴趣的读者可查看本站专题:《Java数据结构与算法教程》、《Java操作DOM节点技巧总结》、《Java文件与目录操作技巧汇总》和《Java缓存操作技巧汇总

希望本文所述对大家java程序设计有所帮助。

相关文章

  • idea中cherry pick的用法

    idea中cherry pick的用法

    Cherry-Pick可以将一个分支的某些commit,合并到另一个分支,本文给大家分享idea中cherry pick的用法,感兴趣的朋友跟随小编一起看看吧
    2023-08-08
  • Java垃圾回收器的方法和原理总结

    Java垃圾回收器的方法和原理总结

    本篇文章主要介绍了Java垃圾回收器的方法和原理总结,Java垃圾回收器是Java虚拟机的重要模块,具有一定的参考价值,有兴趣的可以了解一下。
    2016-12-12
  • java统计字符串中重复字符出现次数的方法

    java统计字符串中重复字符出现次数的方法

    这篇文章主要介绍了java统计字符串中重复字符出现次数的方法,涉及java针对字符串的遍历与判断相关操作技巧,需要的朋友可以参考下
    2016-08-08
  • java 引用类型的数据传递的是内存地址实例

    java 引用类型的数据传递的是内存地址实例

    这篇文章主要介绍了java 引用类型的数据传递的是内存地址实例,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • IDEA项目的依赖(pom.xml文件)导入问题及解决

    IDEA项目的依赖(pom.xml文件)导入问题及解决

    这篇文章主要介绍了IDEA项目的依赖(pom.xml文件)导入问题及解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-08-08
  • SpringBoot登录用户权限拦截器

    SpringBoot登录用户权限拦截器

    这篇文章主要介绍了SpringBoot登录用户权限拦截器,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03
  • maven项目在svn中的上传与检出的方法

    maven项目在svn中的上传与检出的方法

    企业开发中经常使用svn来为我们控制代码版本,也经常使用maven来管理项目。下面将介绍一下如何将maven项目上传到svn中,如何将项目从svn中检出,感兴趣的小伙伴们可以参考一下
    2019-02-02
  • java实现省市区三级联动

    java实现省市区三级联动

    这篇文章主要为大家详细介绍了java实现省市区三级联动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • Java之Class.forName()用法案例详解

    Java之Class.forName()用法案例详解

    这篇文章主要介绍了Java之Class.forName()用法案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • Java 顺序表专题解读

    Java 顺序表专题解读

    顺序表,全名顺序存储结构,是线性表的一种。线性表用于存储逻辑关系为“一对一”的数据,顺序表自然也不例外,不仅如此,顺序表对数据物理存储结构也有要求。顺序表存储数据时,会提前申请一整块足够大小的物理空间,然后将数据依次存储起来,存储时数据元素间不留缝隙
    2021-11-11

最新评论