java 通过解析字符串数学表达式简单进行计算(包括自定义函数以及带括号的数学表达式)
package com.jxv.common.utils; import com.alibaba.fastjson.JSON;import org.apache.commons.lang3.StringUtils;import org.slf4j.Logger;import org.slf4j.LoggerFactory; import javax.script.ScriptEngine;import javax.script.ScriptEngineManager;import javax.script.ScriptException;import java.math.BigDecimal;import java.math.RoundingMode;import java.util.*; import static com.jxv.common.utils.MathCalculatorUtil.SelfMathFormulaEnum.getSelfMathFormulaEnum;import static com.jxv.common.utils.MathCalculatorUtil.SelfMathFormulaEnum.getSelfMathFormulaNames; /** * @Author fangzhenxun * @Description 数学计算公式(精确) * @Date 2020/12/30 13:30 **/public class MathCalculatorUtil { private static final Logger log = LoggerFactory.getLogger(MathCalculatorUtil.class); /** * 所支持的运算操作符集合, 两元运算符 **/ private static final Set<Character> operateSet = new HashSet<>(); /** * 初始化 **/ static { //加 operateSet.add('+'); //减 operateSet.add('-'); //乘 operateSet.add('*'); //除 operateSet.add('/'); //求余 operateSet.add('%'); } /** * 自定义数学公式枚举 **/ enum SelfMathFormulaEnum { abs("abs", 1, 3, "abs(x)", "返回数的绝对值"), acos("acos", 1, 4,"acos(x)", "返回数的反余弦值"), asin("asin", 1, 4,"asin(x)", "返回数的反正弦值"), atan("atan", 1, 4,"atan(x)", "以介于 -PI/2 与 PI/2 弧度之间的数值来返回 x 的反正切值"), ceil("ceil", 1, 4,"ceil(x)", "对数进行上舍入"), cos("cos", 1, 3,"cos(x)", "返回数的余弦"), exp("exp", 1, 3,"exp(x)", "返回 e 的指数"), floor("floor", 1, 5,"floor(x)", "对数进行下舍入"), log("log", 1, 3,"log(x)", "返回数的自然对数(底为e)"), max("max", 2, 3,"max(x,y)", "返回 x 和 y 中的最高值"), min("min", 2, 3,"min(x,y)", "返回 x 和 y 中的最低值"), pow("pow", 2, 3,"pow(x,y)", "返回 x 的 y 次幂"), round("round", 1, 5,"round(x)", "把数四舍五入为最接近的整数"), sin("sin", 1, 3,"sin(x)", "返回数的正弦"), sqrt("sqrt", 1, 4,"sqrt(x)", "返回数的平方根"), tan("tan", 1, 3,"tan(x)", "返回角的正切"); /** * 公式名称 **/ private String formulaName; /** * 公式参数数量 **/ private Integer formulaArgCount; /** * 公式名称长度 **/ private Integer formulaNameLength; /** * 公式表达式 **/ private String formulaExpresion; /** * 公式描述 **/ private String description; /** * @param formulaName * @param formulaArgCount * @return com.jxv.common.utils.MathCalculatorUtil.SelfMathFormulaEnum * @Author fangzhenxun * @Description 根据自定义公式名称,和参数数量返回匹配的枚举实体 * @Date 2020/12/31 10:14 **/ public static SelfMathFormulaEnum getSelfMathFormulaEnum(String formulaName, Integer formulaArgCount) { for (SelfMathFormulaEnum selfMathFormulaEnum : SelfMathFormulaEnum.values()) { if (selfMathFormulaEnum.getFormulaName().equals(formulaName) && selfMathFormulaEnum.getFormulaArgCount().equals(formulaArgCount)) { return selfMathFormulaEnum; } } return null; } /** * @Author fangzhenxun * @Description 根据名称获取函数名 * @Date 2020/12/31 17:10 * @param formulaName * @return com.jxv.common.utils.MathCalculatorUtil.SelfMathFormulaEnum **/ public static SelfMathFormulaEnum getSelfMathFormulaEnum(String formulaName) { for (SelfMathFormulaEnum selfMathFormulaEnum : SelfMathFormulaEnum.values()) { if (selfMathFormulaEnum.getFormulaName().equals(formulaName)) { return selfMathFormulaEnum; } } return null; } /** * @param * @return java.util.List<java.lang.String> * @Author fangzhenxun * @Description 获取自定义公式的简单名称集合 * @Date 2020/12/31 14:44 **/ public static List<String> getSelfMathFormulaNames() { List<String> formulaNames = new ArrayList<>(); for (SelfMathFormulaEnum selfMathFormulaEnum : SelfMathFormulaEnum.values()) { formulaNames.add(selfMathFormulaEnum.getFormulaName()); } return formulaNames; } /** * @Author fangzhenxun * @Description 获取所有的自定义函数枚举 * @Date 2021/1/6 10:27 * @param * @return java.util.List<com.jxv.common.utils.MathCalculatorUtil.SelfMathFormulaEnum> **/ public static List<SelfMathFormulaEnum> getSelfMathFormulas() { List<SelfMathFormulaEnum> formulaNames = new ArrayList<>(); for (SelfMathFormulaEnum selfMathFormulaEnum : SelfMathFormulaEnum.values()) { formulaNames.add(selfMathFormulaEnum); } return formulaNames; } SelfMathFormulaEnum(String formulaName, Integer formulaArgCount, Integer formulaNameLength, String formulaExpresion, String description) { this.formulaName = formulaName; this.formulaArgCount = formulaArgCount; this.formulaNameLength = formulaNameLength; this.formulaExpresion = formulaExpresion; this.description = description; } public Integer getFormulaNameLength() { return formulaNameLength; } public void setFormulaNameLength(Integer formulaNameLength) { this.formulaNameLength = formulaNameLength; } public String getFormulaName() { return formulaName; } public void setFormulaName(String formulaName) { this.formulaName = formulaName; } public Integer getFormulaArgCount() { return formulaArgCount; } public void setFormulaArgCount(Integer formulaArgCount) { this.formulaArgCount = formulaArgCount; } public String getFormulaExpresion() { return formulaExpresion; } public void setFormulaExpresion(String formulaExpresion) { this.formulaExpresion = formulaExpresion; } public String getDescription() { return description; } public void setDescription(String description) { this.description = description; } } /** * JavaScript脚本引擎,Java SE 6开始支持 **/ private static final ScriptEngine scriptEngine = new ScriptEngineManager().getEngineByName("JavaScript"); /** * @param str * @return boolean * @Author fangzhenxun * @Description 判断字符串是否为数字(浮点类型也包括, 以及正负符号) * @Date 2020/12/25 18:18 **/ public static boolean isNumber(String str) { String reg = "^[-\\+]?[0-9]+(.[0-9]+)?$"; return str.matches(reg); } /** * @param mathFormulaScript 数学公式字符串,如:mathFormula = (2*3-45/5+(9)+9%5 +2*(1+2) + Math.sqrt(3))/9.0 * 注意:如果使用开根号等三角函数等一些高级计算,则使用Math中字符串来替代,如4开根号==>Math.sqrt(4) * 具体使用查看:https://www.w3school.com.cn/jsref/jsref_obj_math.asp * @param retainDigit 保留几位小数 * @return java.lang.String * @Author fangzhenxun * @Description 简单公式计算,会有精度丢失问题 * @Date 2020/12/23 13:13 **/ public static String simpleFormulaScript(String mathFormulaScript, int retainDigit) { try { if (StringUtils.isNotEmpty(mathFormulaScript)) { mathFormulaScript = "(" + mathFormulaScript + ").toFixed(" + retainDigit + ")"; } return JSON.toJSONString(scriptEngine.eval(mathFormulaScript)); } catch (ScriptException e) { log.error("非法数学公式!"); e.printStackTrace(); throw new RuntimeException("非法数学公式!"); } } /** * @param mathFormulaScript * @return java.lang.String * @Author fangzhenxun * @Description 重载方法,默认进度小数保留两位 * @Date 2020/12/23 13:48 **/ public static String simpleFormulaScript(String mathFormulaScript) { return simpleFormulaScript(mathFormulaScript, 2); } /** * @param arg * @return void * @Author fangzhenxun * @Description 参数检查 * @Date 2020/12/30 13:37 **/ private static void checkArg(String arg) { checkArg(arg, true); } /** * @param arg * @return void * @Author fangzhenxun * @Description 参数检查 * @Date 2020/12/30 13:37 **/ private static void checkArg(String arg, boolean isZero) { if (StringUtils.isEmpty(arg) || !isNumber(arg) || (!isZero && (new BigDecimal("0").compareTo(new BigDecimal(arg)) == 0))) { throw new RuntimeException("非法计算参数!"); } } /** * @param expression "1+1*2+(10-(2*(5-3)*(2-1))-4)+10/(5-0) + log(10) + log(10,12)" * @return void * @Author fangzhenxun * @Description 校验公式表达式的合法性 * @Date 2020/12/30 14:23 **/ private static void checkFormulaExpression(String expression) { //去除空格 expression = expression.replaceAll(" ", ""); //拆分字符串 char[] arr = expression.toCharArray(); int len = arr.length; //前后括号计数,用来判断括号是否合法 int checkNum = 0; //数字集合 StringBuffer sb = new StringBuffer(); //字母集合 StringBuffer sb0 = new StringBuffer(); //循环 for (int i = 0; i < len; i++) { //判断当前元素是否是数字 if (Character.isDigit(arr[i]) || arr[i] == '.') { //把数字和小数点加入到集合中,为了下一步判断数字是否合法 sb.append(arr[i]); } else if (Character.isLetter(arr[i])) { //校验自定义的公式(自定义的数学表达式都是使用字母拼接起来) sb0.append(arr[i]); } else { //如果sb中有值,取出来判断这个数字整体是否合法 if (sb.length() > 0) { if (isNumber(sb.toString())) { //如果合法,清空,为了判断下一个 sb.setLength(0); } else { throw new RuntimeException("非法数字参数"); } } //不是数字为字符,可能是{},[], (), +, - , * , /, % 等,再加上各种自定义的数学运算公式, 或者是其他不合法的字符,接着继续判断 if (arr[i] == '+' || arr[i] == '*' || arr[i] == '/' || arr[i] == '%') { //判断规则(1.不能位于首位 2.不能位于末尾 3.后面不能有其他运算符,但可以有-号 4.后面不能有后括号) if (i == 0 || i == (len - 1) || arr[i + 1] == '+' || arr[i + 1] == '*' || arr[i + 1] == '/' || arr[i + 1] == '%' || arr[i + 1] == ')') { log.error("非法符号 : '+' or '*' or '/' ->" + arr[i]); throw new RuntimeException("非法符号 : '+' or '*' or '/' ==>" + arr[i]); } } else if (arr[i] == '-') { //减号判断规则(1.不能位于末尾 2.后面不能有其他运算符,但可以有-号 3.后面不能有后括号) if (i == (len - 1) || arr[i + 1] == '+' || arr[i + 1] == '*' || arr[i + 1] == '/' || arr[i + 1] == '%' || arr[i + 1] == ')') { log.error("非法符号 : '-' ->" + arr[i]); throw new RuntimeException("非法符号 : '-' ==>" + arr[i]); } } else if (arr[i] == '(') { //判断(括号前面是否有字母 //如果sb0中有字母,取出来判断这个字符串整体是否是自定义的公式字符 if (sb0.length() > 0) { //从当前匹配的(,找到最近的),然后在求出在 这两个括号之间以英文逗号隔开的参数个数 int beginIndex = expression.indexOf(arr[i], i); int endIndex = matchBracketIndex(expression, i, arr[i]); if (endIndex == -1) { log.error("非法数学公式符号:==>" + sb0.length()); throw new RuntimeException("非法数学公式符号: ==>" + sb0.length()); } //截取字符串,且分隔匹配的英文逗号 String selfMathBracketContentStr = expression.substring(beginIndex + 1, endIndex); if (StringUtils.isEmpty(selfMathBracketContentStr)) { log.error("非法自定义数学公式符号:==>" + sb0.length()); throw new RuntimeException("非法自定义数学公式符号: ==>" + sb0.length()); } //获取参数个数 StringBuilder selfMathBracketContentSb = new StringBuilder(selfMathBracketContentStr); int argCounts = getSelfMathMarkArgCounts(selfMathBracketContentSb, ","); //校验自定义公式的合法性 checkSelfMathMark(sb0.toString(), argCounts); //清空内容 sb0.setLength(0); } checkNum++; //判断规则(1.不能位于末尾 2.后面不能有+,*,/,%运算符和后括号 3.前面不能为数字) if (i == (len - 1) || arr[i + 1] == '+' || arr[i + 1] == '*' || arr[i + 1] == '/' || arr[i + 1] == '%' || arr[i + 1] == ')' || (i != 0 && Character.isDigit(arr[i - 1]))) { log.error("非法符号 : '(' ->" + arr[i]); throw new RuntimeException("非法符号 : '(' ==>" + arr[i]); } } else if (arr[i] == ')') { checkNum--; //判定规则(1.不能位于首位 2.后面不能是前括号 3.括号计数不能小于0,小于0说明前面少了前括号) if (i == 0 || (i < (len - 1) && arr[i + 1] == '(') || checkNum < 0) { log.error("非法符号 : ')' ->" + arr[i]); throw new RuntimeException("非法符号 : ')' ==>" + arr[i]); } } else if (arr[i] == ',') { //判定规则,如果有逗号,1,匹配该逗号是否被括号()包着,且左括号当前前面是否是自定义公式 checkComma(expression, i, i); } else { //非数字和运算符 log.error("非数字和运算符:==>" + arr[i]); throw new RuntimeException("非数字和运算符:==>" + arr[i]); } } } //不为0,说明括号不对等,可能多前括号 if (checkNum != 0) { //非数字和运算符 log.error("括号个数不匹配"); throw new RuntimeException("括号个数不匹配"); } } /** * @param str 待匹配的字符数组 * @param currentCommaIndex 当前逗号索引 * @param constCommaIndex 常量逗号索引 * @return void * @Author fangzhenxun * @Description 校验英文逗号 * @Date 2020/12/31 13:24 **/ private static void checkComma(String str, int currentCommaIndex, final int constCommaIndex) { //从currentCommaIndex索引开始往前找最近的一个左括号(,求出索引,并求出对应的)索引 int beginIndex = indexOfBefore(str, currentCommaIndex, '('); if (beginIndex == -1) { log.error("非法逗号!"); throw new RuntimeException("非法逗号!"); } int endIndex = matchBracketIndex(str, beginIndex, '('); if (endIndex == -1) { log.error("非法逗号!"); throw new RuntimeException("非法逗号!"); } //找到符合的逗号条件 if (endIndex <= constCommaIndex) { //接着往上找 checkComma(str, beginIndex, constCommaIndex); } else { //在从beginIndex索引开始往前找出连续字母的字符,然后拼接 char[] chars = str.toCharArray(); StringBuilder sb = new StringBuilder(); for (int i = beginIndex - 1; i >= 0; i--) { //如果是字母或者是数字 if (Character.isLetter(chars[i]) || Character.isDigit(chars[i])) { sb.append(chars[i]); } else { break; } } //将sb带入自定义公式进行校验 List<String> selfMathFormulaNames = getSelfMathFormulaNames(); if (!selfMathFormulaNames.contains(sb.reverse().toString())) { throw new RuntimeException("非法逗号!"); } } } /** * @param str 目标字符串 如 abcd((a),(b,bb))dada * @param endIndex * @return int * @Author fangzhenxun * @Description 返回从结束索引结束, * @Date 2020/12/31 13:36 **/ private static int indexOfBefore(String str, int endIndex, char dest) { char[] chars = str.trim().toCharArray(); int len = chars.length; int index0 = endIndex; if (len - 1 < endIndex) { index0 = len; } for (int i = index0 - 1; i >= 0; i--) { if (chars[i] == dest) { return i; } } //没有找到返回-1 return -1; } /** * @param mathArgStr 参数括号字符串 如:log((math(1,6)+2),(8*9)) 中的 ((math(1,6)+2),(8*9)) * @param regex 分隔符 默认一般英文逗号, * @return void * @Author fangzhenxun * @Description 获取自定义公式的参数个说 * @Date 2020/12/31 10:30 **/ private static int getSelfMathMarkArgCounts(StringBuilder mathArgStr, String regex) { //找到最近的一个( 索引 int beginIndex = mathArgStr.indexOf("("); if (beginIndex != -1) { int endIndex = matchBracketIndex(mathArgStr.toString(), beginIndex, '('); if (endIndex == -1) { throw new RuntimeException("非法括号匹配!"); } mathArgStr.replace(beginIndex, endIndex + 1, ""); return getSelfMathMarkArgCounts(mathArgStr, regex); } else { //如果没有匹配的做扩号,则直接求出参数个数 String[] argsArr = mathArgStr.toString().split(regex, -1); return argsArr.length; } } /** * @param mathStr log sin * @param argCount 2 , 1 * @return boolean * @Author fangzhenxun * @Description 校验数字字符数组 * @Date 2020/12/30 15:09 **/ private static void checkSelfMathMark(String mathStr, int argCount) { SelfMathFormulaEnum selfMathFormulaEnum = getSelfMathFormulaEnum(mathStr, argCount); if (selfMathFormulaEnum == null) { throw new RuntimeException("自定义数学公式不匹配!"); } } /** * @param s 待匹配的字符串 * @param fromIndex 开始索引 * @param leftDest 左括号 * @return int * @Author fangzhenxun * @Description 通过当前左括号(的索引 , 找到与之匹配对应的右括号) 的索引 * @Date 2020/12/30 17:23 **/ private static int matchBracketIndex(String s, int fromIndex, char leftDest) { if (StringUtils.isEmpty(s)) { return -1; } //取出匹配目标的第一个索引 int index0 = s.indexOf(leftDest, fromIndex); if (index0 != -1) { //1、申明一个stack Stack<Character> stack = new Stack<>(); //遍历s String本质上是char[] for (int i = index0; i < s.length(); i++) { char c = s.charAt(i); if (c == '{' || c == '[' || c == '(') { //如果是{ [ ( 压入栈中 stack.push(c); } else if (c == '}' || c == ']' || c == ')') { // } ] ) 进行比对, if (stack.isEmpty()) { return -1; } char topChar = stack.pop(); if ((topChar == '[' && c == ']') || (topChar == '(' && c == ')') || (topChar == '{') && c == '}') { if (stack.isEmpty()) { return i; } else { continue; } } } else { continue; } } } return -1; } /** * @param v1 * @param v2 * @return java.lang.String * @Author fangzhenxun * @Description 两个数相加 * @Date 2020/12/30 13:42 **/ private static String add(String v1, String v2) { //校验参数 checkArg(v1); checkArg(v2); BigDecimal v1Bd = new BigDecimal(v1); BigDecimal v2Bd = new BigDecimal(v2); return v1Bd.add(v2Bd).toString(); } /** * @param v1 * @param v2 * @return java.lang.String * @Author fangzhenxun * @Description 两个数相减 * @Date 2020/12/30 13:47 **/ private static String sub(String v1, String v2) { //校验参数 checkArg(v1); checkArg(v2); BigDecimal v1Bd = new BigDecimal(v1); BigDecimal v2Bd = new BigDecimal(v2); return v1Bd.subtract(v2Bd).toString(); } /** * @param v1 * @param v2 * @return java.lang.String * @Author fangzhenxun * @Description 两个数向乘 * @Date 2020/12/30 13:49 **/ private static String mul(String v1, String v2) { //校验参数 checkArg(v1); checkArg(v2); BigDecimal v1Bd = new BigDecimal(v1); BigDecimal v2Bd = new BigDecimal(v2); return v1Bd.multiply(v2Bd).toString(); } /** * @param v1 * @param v2 * @return java.lang.String * @Author fangzhenxun * @Description 两个数相除 * @Date 2020/12/30 13:50 **/ private static String div(String v1, String v2) { //校验参数 checkArg(v1); //除数不能为0 checkArg(v2, false); BigDecimal v1Bd = new BigDecimal(v1); BigDecimal v2Bd = new BigDecimal(v2); return v1Bd.divide(v2Bd, 2, RoundingMode.HALF_UP).toString(); } /** * @Author fangzhenxun * @Description v1%v2 取余 * @Date 2021/1/4 22:14 * @param v1 * @param v2 * @return java.lang.String **/ private static String mod(String v1, String v2) { //校验参数 checkArg(v1); //除数不能为0 checkArg(v2, false); BigDecimal v1Bd = new BigDecimal(v1); BigDecimal v2Bd = new BigDecimal(v2); return v1Bd.remainder(v2Bd).toString(); } /** * @param mathFormula 公式字符串, 如:"1+1*2+(10-(2*(5-3)*(2-1))-4)+10/(5-0) + log(10) + log(10,12)" * 注意log(10) ==> ln(10) ==> log(e,10) 表示以自然数e为底的对数 * @return java.lang.String * @Author fangzhenxun * @Description 使用递归计算,判断表达式是否有括号,有括号,则先计算括号内的表达式,无则直接运算结果。 * @Date 2020/12/30 14:07 **/ public static String calculator(String mathFormula) { if (StringUtils.isEmpty(mathFormula)) { throw new RuntimeException("非法计算公式!"); } //替换空格 mathFormula = mathFormula.replaceAll(" ", ""); int bracket = mathFormula.indexOf("["); int brace = mathFormula.indexOf("{"); if (bracket != -1 || brace != -1) { //将字符串中的"{}"、"[]"替换成"()" log.info("计算公式:{}", mathFormula); mathFormula = mathFormula.replaceAll("[\\[\\{]", "(").replaceAll("[\\]\\}]", ")"); log.info("标准数学计算公式 '{,[':" + mathFormula); } //校验公式参数是否合法 checkFormulaExpression(mathFormula); //==================================================开始计算============================================= //计算思路:以下是计算顺序 // 1,如果有自定义的数学公式,则先计算自定义的公式 // 2, 如果有括号,则先计算括号内的(去括号) // 3, 没有括号直接计算 String result0 = calculatorSelfMathFormula(mathFormula); //结果保留八位小数 return new BigDecimal(result0).setScale(8,BigDecimal.ROUND_HALF_UP).toString(); } //"1+1*2+(10-(2*(5-3)*(2-1))-4)+10/(5-0) + log(10) + max(sin(2)*6,pow(2,3))" public static void main(String args[]) throws ScriptException { long beginTime = System.currentTimeMillis(); String calculator2 = MathCalculatorUtil.standardCalculation("3*-2"); String calculator3 = MathCalculatorUtil.standardCalculation("3--2"); String k ="-(2.5)*(-1)+(-1)*2"; String k1 ="-2.5*(2)+(-1)*(-2)"; String k2 ="(-2.5)*(-2)+(-1)*(-2)-2*3"; String re2 = calculator(k2); String ss22 = "1+1*2+(10-(2*(5-3)*(2-1))-4)+10/(5-0) + log(10)/9 + 2*log(30)+ max(sin(2*(5+2))*6,pow(2*(2+8),3+2)) * max(-1+3,(5-4)) + min(sin(10), sin(20))"; String re0 = calculator(k); String re1 = calculator(k1); System.out.println(re0);// Object eval = scriptEngine.eval(ss23);// System.out.println(eval); System.out.println("cost时间:" + (System.currentTimeMillis() - beginTime) + "ms"); } /** * @Author fangzhenxun * @Description 去除自定义公式,todo 待优化 * @Date 2020/12/31 16:49 * @param mathFormula * @return java.lang.String **/ private static String calculatorSelfMathFormula(String mathFormula) { if (StringUtils.isEmpty(mathFormula)) { throw new RuntimeException("非法参数错误!"); } mathFormula = mathFormula.replaceAll(" ", ""); //去除自定义公式 List<SelfMathFormulaEnum> selfMathFormulaEnums = SelfMathFormulaEnum.getSelfMathFormulas(); boolean flag = false; for (SelfMathFormulaEnum mathFormulaEnum : selfMathFormulaEnums) { if (mathFormula.contains(mathFormulaEnum.getFormulaName())) { flag = true; break; } } //包含自定义公式 if (flag) { for (int i = 0; i < selfMathFormulaEnums.size();) { boolean repeat = false; SelfMathFormulaEnum mathFormulaEnum = selfMathFormulaEnums.get(i); //如果该公式表达式包含自定义数学公式 "1+1*2+(10-(2*(5-3)*(2-1))-4)+10/(5-0) + log(10) + 2*log(30)+ max(sin(2*(5+2))*6,pow(2*(2+8),3+2)) * max(1+3,(3-5)) + min(sin(10), sin(20))" if (mathFormula.contains(mathFormulaEnum.getFormulaName())) { //如果匹配到,则获取第一个自定义的数学公式首字母所在的索引(该索引是格式化后的索引) int index0 = mathFormula.indexOf(mathFormulaEnum.getFormulaName()); //取出该公式括号中内容字符,不包括左右字符 String left = mathFormula.substring(0, index0); int index1 = matchBracketIndex(mathFormula, index0, '('); String right = mathFormula.substring(index1 +1); String bracketsContent = mathFormula.substring(index0 + mathFormulaEnum.getFormulaNameLength() + 1, index1); //计算括号中的值,如果该字符串又包含其他自定义公式,则递归继续计算 //left + result0 + right mathFormula = left + selfMathCalculation(mathFormulaEnum.getFormulaName(), calculatorSelfMathFormula(bracketsContent)) + right; repeat = true; } if (repeat) { i = i; }else { i++; } } } //直接进行计算 return standardCalculation(mathFormula); } /** * @Author fangzhenxun * @Description 自定义公式计算 * @Date 2020/12/31 17:07 * @param mathFormulaName 自定义公式名称 * @param digitStr 一个具体的数值 * @return java.lang.String **/ private static String selfMathCalculation(String mathFormulaName, String digitStr) { double result; if (StringUtils.isEmpty(digitStr)) { throw new RuntimeException("非法计算公式参数!"); } String[] args = digitStr.split(",", -1); SelfMathFormulaEnum selfMathFormulaEnum = getSelfMathFormulaEnum(mathFormulaName); if (selfMathFormulaEnum == null) { throw new RuntimeException("非法数学公式名称"); } switch (selfMathFormulaEnum) { case abs: result = Math.abs(Double.parseDouble(args[0])); break; case acos: result = Math.acos(Double.parseDouble(args[0])); break; case asin: result = Math.asin(Double.parseDouble(args[0])); break; case atan: result = Math.atan(Double.parseDouble(args[0])); break; case ceil: result = Math.ceil(Double.parseDouble(args[0])); break; case cos: result = Math.cos(Double.parseDouble(args[0])); break; case exp: result = Math.exp(Double.parseDouble(args[0])); break; case floor: result = Math.floor(Double.parseDouble(args[0])); break; case log: result = Math.log(Double.parseDouble(args[0])); break; case max: result = Math.max(Double.parseDouble(args[0]), Double.parseDouble(args[1])); break; case min: result = Math.min(Double.parseDouble(args[0]), Double.parseDouble(args[1])); break; case pow: result = Math.pow(Double.parseDouble(args[0]), Double.parseDouble(args[1])); break; case round: result = Math.round(Double.parseDouble(args[0])); break; case sin: result = Math.sin(Double.parseDouble(args[0])); break; case sqrt: result = Math.sqrt(Double.parseDouble(args[0])); break; case tan: result = Math.tan(Double.parseDouble(args[0])); break; default: throw new RuntimeException("找不到匹配的计算公式!"); } return String.valueOf(result); } /** * @Author fangzhenxun * @Description 标准计算,不包含自定义函数, 但包含括号与其他符号表达式 * @Date 2020/12/31 17:02 * @param str * @return java.lang.String **/ private static String standardCalculation(String str) { if (StringUtils.isEmpty(str)) { log.error("非法计算公式!"); throw new RuntimeException("非法计算公式!"); } String[] args = str.split(",", -1); if (args != null && args.length > 0) { List<String> argResult = new ArrayList<>(); for (String arg : args) { //每一个arg 都是一个算式(带上括号的) //判断是公式表达是是否存在小括号(优先级) int hasBrackets = arg.lastIndexOf('('); if (hasBrackets == -1) { //没有小括号,直接计算 argResult.add(cac(arg)); }else { int cr = arg.indexOf(')', hasBrackets); String left = arg.substring(0, hasBrackets); String right = arg.substring(cr + 1); //如果存在"("提取括号中的表达式 String middle = arg.substring(hasBrackets + 1, cr); argResult.add(standardCalculation(left + cac(middle) + right)); } } return StringUtils.join(argResult, ","); } throw new RuntimeException("非法算式参数!"); } /** * DESC:计算表达式,判断是否存在乘除运算,存在则先执行乘除运算,然后执行加减运算,返回运算结果; * 不存在,直接运行加减运算,返回运算结果。 * * @param str -2.8+8*3/2+0.9 或 -2.8*6 * @return 运算结果 */ private static String cac(String str) { //字符串中不存在*,/, % int mulIndex = str.indexOf('*'); int divIndex = str.indexOf('/'); int modIndex = str.indexOf('%'); //只有加法和减法 if (mulIndex == -1 && divIndex == -1 && modIndex == -1) { return AASOperation(str); } String result0 = "0"; //定义先处理的符号索引位置 int index0 = getMin(-1,mulIndex, divIndex, modIndex); try { String left = str.substring(0, index0); String v1 = lastNumber(left); left = left.substring(0, left.length() - v1.length()); String right = str.substring(index0 + 1); String v2 = firstNumber(right); right = right.substring(v2.length()); if (index0 == mulIndex) { result0 = mul(v1, v2); } else if(index0 == divIndex) { result0 = div(v1, v2); } else if(index0 == modIndex) { result0 = mod(v1, v2); } String s = left + result0 + right; return cac(left + result0 + right); }catch (Exception e) { log.error("数学计算公式错误"+ e.getMessage()); throw new RuntimeException("数学计算公式错误!"); } } /** * @Author fangzhenxun * @Description 求给定可变参数中不等于noNum 的最小值 * @Date 2021/1/5 10:49 * @param noNum * @param a 可变参数 * @return int **/ private static int getMin(int noNum, int... a){ if (a == null || a.length == 0) { throw new RuntimeException("非法参数!"); } int min = a[0]; for (int i = 1; i < a.length; i++) { if (min ==noNum || (min > a[i] && a[i] != noNum)) { min = a[i]; } } if (min == noNum) { throw new RuntimeException("非法可变参数!"); } return min; } /** * @Author fangzhenxun * @Description 获得表达式的最后连续合法数字 * @date 2021/1/6 17:49 * @param str * @return java.lang.String */ private static String lastNumber(String str) { StringBuilder sb = new StringBuilder(); for (int i = str.length() - 1; i >= 0; i--) { char c = str.charAt(i); //包含首字母为- if (Character.isDigit(c) || (i != 0 && c == '.') || ((i == 0 || operateSet.contains(str.charAt(i -1))) && c == '-')) { sb.append(c); }else { break; } } return sb.reverse().toString(); } /** * @Author fangzhenxun * @Description 获得表达式的最后连续合法数字 * @date 2021/1/6 17:49 * @param str * @return java.lang.String */ private static String firstNumber(String str) { StringBuilder sb = new StringBuilder(); for (int i = 0; i < str.length(); i++) { char c = str.charAt(i); //包含首字母为- if (Character.isDigit(c) || (i != 0 && c == '.') || (i == 0 && c == '-')) { sb.append(c); }else { break; } } return sb.toString(); } /** * @Author fangzhenxun * @Description 只用加减操作 * @Date 2021/1/4 20:02 * @param mathStr 只有加减操作的数学运算字符串: 如2.98-5-6+9-0.2-8 或 -2.0-9-5+9 或 -9-2 * @return java.lang.String **/ private static String AASOperation(String mathStr) { if (StringUtils.isEmpty(mathStr)) { throw new RuntimeException("非法计算参数"); } //这里字符串加上一个运算法号,只要是合法的都可以,只是为了走一步运算 char[] options = (mathStr + "+").replaceAll(" ", "").toCharArray(); String result0 = "0"; StringBuilder sb = new StringBuilder(); char sign = '+'; for (int i = 0; i < options.length; i++) { if (Character.isDigit(options[i]) || options[i] == '.') { sb.append(options[i]); } else { if ((i == 0 && options[i] == '-') || (i>1 && operateSet.contains(options[i-1]))) { sb.append(options[i]); }else { if (sb.length() > 0){ //先默认为 + 把第一个数值加上 if (sign == '+') { result0 = add(result0, sb.toString()); } else { result0 = sub(result0, sb.toString()); } sb.setLength(0); sign = options[i]; } else { throw new RuntimeException("非法数学公式错误!"); } } } } return result0; }}
赞 (0)