最近给公司开发业务代码时,碰到一个场景,简单描述是这样的:

  客户要向咱们公司定制一件产品,这个产品呢,有很多属性,那公司得根据这些属性报价呀,怎么报价呢?公司针对某种类型的产品有一个基准价,在同类产品下,某个属性超标了,需要加价,但每一个属性的加价方式都不一样,针对每一家客户加多少价也不一样,每个时间点加价比率也可能不一样,真实情况要比这个复杂不少,这里就不再深入讨论。

  那么应对这种需求,我首先想到的关键点是:要把加价这个公式,暴露给实际能控制它的人员去输入,把公式中需要用到的一些参数,以替代符(或者说变量)的方式提供给他们,比如,用a表示基准价,b表示属性超出数值,然后超出部分需要乘以该属性的单价5块钱,那么最终的值就可以写成公式: a + b * 5 ;实际运算的时候,假设a是100,b是20,把他们代替 a和b,公式就成了 100 + 20 * 5,看起来很简单的公式,口算都能算出来,但是正常来讲,公式录入系统,是以字符串的形式保存的,一直到你把真实的值替换到公式里,也是字符串操作,计算机要如何把你这字符串的里的内容正确的计算出来呢?

  OK,其实对代码逻辑不是非常好的同学,可以用一些简单的方法,比如,将最终的公式用JavaScript的eval()函数执行一下,就可以得到结果了,这个方法也可以用来在前端验证公式录入正确与否,还有一种方法,把最终的公式直接拼接到SQL语句,对上临时表查一下,如:SELECT  100+20*5  AS  result from dual;也可以得到结果,用这两种方法,其实还可能进行更复杂的计算,充分利用JavaScript和sql提供的函数库。

  嗯,回到原点,我们现在呢,要在Java服务端实现字符串的公式正确计算,公式虽然简单,但可能每一次要进行几百条公式的计算,也没必要查询几百次数据库,而且这种和金钱相关联的值,如非必要,还是不要抛给前端来替你计算。虽然,公司这边并没有采用我的方案,但是我个人还是把一个简单的公式计算器写了出来,留个思路,以作备用。

  简单公式计算器能够满足 加减乘除 和 小括号的运算。

  我个人非常建议新手练习一下,基础运用得越扎实,对以后的技术瓶颈突破越好。

  代码图上的注释比较少:

  粘贴出代码图:

代码来了:

package com.supalle.test;

import java.math.BigDecimal;
import java.math.MathContext; /**
* @作者: Supalle
* @时间: 2019/3/8
* @描述: 简单公式计算器
*/
public class Calc { private char[] val; private int len; private int inx; // 构造器,把公式传进去,比如: 100 + 20 * 5 + (1 + 2)
public Calc(String val) {
this.val = val.toCharArray();
len = this.val.length;
inx = 0;
} // 获取计算结果,使用方法其实就是 new Calc("100 + 20 * 5 + (1 + 2)").getResult();就可以得到结果了
public BigDecimal getResult() {
return nextValue(BigDecimal.ZERO, '+');
}

   // OK,接下来的两个方法,必须要弄明白,下一个值和下一个参数的区别
   // 为什么要获取下一个值,加法、减法、和左小阔号,都需要获取下一个值,因为加法、减法如果碰到乘法、除法,那么运算优先权在右侧,如果碰到左侧小括号,优先权也在右侧,所以要先把右边的值算出来
   // 为什么要获取下一个参数,乘法、除法,他们下一个运算符如果不是左侧小括号,那么应该从左往右顺序计算,因此需要直接取到下一个参数进行计算
   // 还有一点要值得注意,那就是:在运算时,减法一律替换成加上一个负数,以此来消除实际对一个负数进行运算产生异常,比如 1 * -3,总不能检测到 - 的时候,又去做减法运算吧
   // 就讲这么多了,不能理解的同学,再反复推敲几遍
// 获取下一个值,传入第一个参数和第一个参数后的运算符
private BigDecimal nextValue(BigDecimal param1, char operator) {
if (inx < len) {
if (operator == ')') {
return param1;
} if (operator == '+') {
return param1.add(nextValue(nextParam(), inx < len ? val[inx++] : ')'));
} else if (operator == '*') {
return nextValue(param1.multiply(nextParam(), MathContext.DECIMAL128), inx < len ? val[inx++] : ')');
} else if (operator == '/') {
return nextValue(param1.divide(nextParam(), MathContext.DECIMAL128), inx < len ? val[inx++] : ')');
}
}
return param1;
} // 获取下一个参数
private BigDecimal nextParam() { char[] param = new char[len - inx + 1]; int paramInx = 0; while (inx < len) { if (val[inx] == '-') {
if (paramInx == 0) {
param[paramInx++] = val[inx];
param[paramInx++] = '0';
} else {
val[--inx] = '+';
break;
}
} else if (val[inx] == '.' || ((int) val[inx] >= 48 && (int) val[inx] <= 57)) {// 如果是 . 或 0 ~ 9
param[paramInx++] = val[inx];
} else if (val[inx] == '(') {
inx++;
return nextValue(BigDecimal.ZERO, '+');
} else if (((int) val[inx] >= 41 && (int) val[inx] <= 43) || (int) val[inx] == 47) {
break;
} inx++;
} return paramInx > 0 ? new BigDecimal(param, 0, paramInx) : BigDecimal.ZERO;
} }

Java简单公式计算器的更多相关文章

  1. Java使用BigDecimal精确计算的简单公式计算器

    由于工作需要,写了一个使用BigDecimal运算的精确计算的计算器(然后发现其实比不用BigDecimal的并好不到哪里去) 只能做加减乘除 double类型的数字在千万级别的时候会转成科学计数法, ...

  2. jsp学习---使用jsp和JavaBean实现超简单网页计算器

    一.需求 如题,用jsp实现一个超简单的网页计算器. 二.实现 1.效果图 1)初始界面: 2)随便输入两个数进行相乘: 3)当除数为零时提示报错: 2.代码 Calculator.java pack ...

  3. 教学项目之-通过Python实现简单的计算器

    教学项目之-通过Python实现简单的计算器   计算器开发需求 实现加减乘除及拓号优先级解析 用户输入 1 - 2 * ( (60-30 +(-40/5) * (9-2*5/3 + 7 /3*99/ ...

  4. HDU1237 简单的计算器 【堆】+【逆波兰式】

    简单的计算器 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Sub ...

  5. 多元线性回归----Java简单实现

    http://www.cnblogs.com/wzm-xu/p/4062266.html 多元线性回归----Java简单实现   学习Andrew N.g的机器学习课程之后的简单实现. 课程地址:h ...

  6. 从0到1:使用Caliburn.Micro(WPF和MVVM)开发简单的计算器

    从0到1:使用Caliburn.Micro(WPF和MVVM)开发简单的计算器 之前时间一直在使用Caliburn.Micro这种应用了MVVM模式的WPF框架做开发,是时候总结一下了. Calibu ...

  7. js制作简单的计算器

    学着做了一个简单的计算器!记录记录!哈哈 <!DOCTYPE html> <html> <head> <title>简单的计算器</title&g ...

  8. java简单词法分析器(源码下载)

    java简单词法分析器 : http://files.cnblogs.com/files/hujunzheng/%E7%AE%80%E5%8D%95%E8%AF%8D%E6%B3%95%E5%88%8 ...

  9. 留念 C语言第一课简单的计算器制作

    留念 C语言第一课简单的计算器制作 学C语言这么久了.  /* 留念 C语言第一课简单的计算器制作 */   #include<stdio.h>  #include<stdlib.h ...

随机推荐

  1. 安装Cloudera Impala

    安装Cloudera Impala Cloudera Impala是Cloudera Enterprise Core的开源扩展,用于快速返回查询结果. Impala作为你环境的插件,与其他组件的安装独 ...

  2. Hadoop集群(第2期)虚拟机网卡的介绍和配置

    很多人安装虚拟机的时候,经常遇到不能上网的问题,而vmware有三种网络模式,对初学者来说也比较眼花聊乱,今天我就来基于虚拟机3种网络模式,帮大家普及下虚拟机上网的背景知识 虚拟机网络模式 无论是vm ...

  3. PDF Expert for Mac v2.4.22 中文破解版下载 PDF阅读编辑软件

    PDF Expert for Mac v2.4.22 中文破解版下载:http://h5ip.cn/CsRN PDF Expert for Mac, iOS 上最好用的 PDF 编辑器之一,现在终于打 ...

  4. 支持向量机 (二): 软间隔 svm 与 核函数

    软间隔最大化(线性不可分类svm) 上一篇求解出来的间隔被称为 "硬间隔(hard margin)",其可以将所有样本点划分正确且都在间隔边界之外,即所有样本点都满足 \(y_{i ...

  5. 小程序请求接口统一封装到一个js文件中

    在我们做小程序时,数据请求数据请求是避免不了的,然而我们用官方自带的请求方式,会给我们带来很多重复的工作,所以我就借鉴大神们的博客,写了一个简单的请求方式. 1.首先我们在utils中新建一个api. ...

  6. 第一章 corejava的入门

    第一章 corejava的入门一:什么是语言语言=os+数据结构+算法+思想os:操作系统数据结构:队,栈,二叉树,链表算法:做游戏开发时非常重要面试题:int a>0,b>0只使用一条输 ...

  7. 【记录】mybatis中获取常量类中数据

    部分转载,已注明来源: 1.mybatis中获取常量类中数据 <update id="refuseDebt"> UPDATE dt_debt a SET         ...

  8. 简单的python多线程实例

    今天抽时间又学了一下python的多线程,理解的又多了一些,为了利于理解,写了一段很简单的代码,如下: import threading from time import sleep def task ...

  9. JDK1.8集合之HashMap

    目录 简介 内部实现 类的属性 Node数组 重要方法 put()和putVal()方法 get()和getNode()方法 resize()方法 容量设置为2的幂的优点 计算Hash时候 扩容时候 ...

  10. RabbitMQ windows 安装后 此时不应有 \RabbitMQ。

    在安装过程中,如果细心实际上在安装的细节显示starting Service后就已经提示这个问题,实际上就是没有注册到RabbitMQ这个服务. 在研究了很久之后才发现原来是安装路径的问题,原先是安装 ...