利用ScriptEngineManager实现字符串公式灵活计算的方法

在开发中我们可能会遇到好多不固定的公式计算 如有时候我们需要将excel中大量的计算公式转换成java语言进行实现

如果说单纯的用java的基本方法实现计算 我们就需要的分析excel中的公式将其一一并转换成java代码 这样对我们开发人员来说工作量有点太大了 而且在转换过程中很可能会出错

下面就介绍一种简单的实现方法

我们知道js的eval()方法可以执行字符串的代码 而恰好jdk6增加了对脚本语言的支持 我们可以利用这个特性对计算实现简单化的处理

下面举个例子

加入有个公式

A+B*C 

A=1,B=2,C=3

我们可以将公式的A B C替换成数字 转换为 1+2*3 最后就可以得到结果了

接下来 我么就探讨下实现方法

直接以代吗的形式来写吧

public Double Calculation(String formula){
Double result=null; //计算结果
ScriptEngineManager manager = new ScriptEngineManager(); //创建一个ScriptEngineManager对象
ScriptEngine engine = manager.getEngineByName("js"); //通过ScriptEngineManager获得ScriptEngine对象
try {
result = (Double) engine.eval(formula); //用ScriptEngine的eval方法执行脚本
} catch (ScriptException e) {
this.isSuccess=false;
result=Double.NaN;
}
return result;
}

利用上面的方法我们就可以将一个数学表达式计算出结果

Calculation(“1+2*3”)-->得到结果7

显然这样是不能实现公式的灵活计算的 怎么实现A+B*C 此时我们应该会想到将参数ABC替换成数字就可以实现计算结果了

接下来我们就探讨如何实现替换的功能

说到替换 我们不难想到String的replaceAll方法,但是这样我们会遇到一个问题就是替换形如A+AA+AAA的问题

如果str.replaceAll("A",2) ;结果就是2+22+222了

所以我想到了利用正则去准确匹配替换 方法如下

参数1:替换公式字的符串 参数2: 公式中要替换的字母如上边的A 参数3:要替换成的数值

/**
 * 精确替换字符 防止出现 匹配A 时将AA匹配的情况
 */

public static String replaceStr(String sourceStr,String replaceKey,String replaceValue){
String replaceStrReg="";
for(char str_char:replaceKey.toCharArray()){
replaceStrReg+="[";
replaceStrReg+=str_char;
replaceStrReg+="]";
}
String startReg="^"+replaceStrReg+"([\\+\\-\\*/,)])";
String endReg="([\\+\\-\\*/,(])"+replaceStrReg+"$";
String reg="([^a-zA-Z])("+replaceStrReg+")"+"([^a-zA-Z])";
String endStr=sourceStr;
while(matcheStr(endStr,replaceKey)){
endStr=endStr.replaceAll(startReg, replaceValue+"$1");
endStr=endStr.replaceAll(reg, "$1"+replaceValue+"$3");
endStr=endStr.replaceAll(endReg, "$1"+replaceValue);
}
return endStr;
}
/**
* 精确匹配字符 防止出现 匹配A 时将AA匹配的情况
*/
public static Boolean matcheStr(String sourceStr,String matchStr){
String replaceStrReg="";
for(char str_char:matchStr.toCharArray()){
replaceStrReg+="[";
replaceStrReg+=str_char;
replaceStrReg+="]";
}
String startReg="^"+replaceStrReg+"([\\+\\-\\*/,)])[\\s\\S]*";
String endReg="[\\s\\S]*([\\+\\-\\*/,(])"+replaceStrReg+"$";
String reg="[\\s\\S]*([^A-Za-z])("+replaceStrReg+")"+"([^A-Za-z])[\\s\\S]*";
if(sourceStr.matches(startReg)||sourceStr.matches(reg)||sourceStr.matches(endReg)){
return true;
}else{
return false;
}
}

上述的正则表达式就是利用了数学公式中出现+-*/运算符号的规律

这样我们就可以先替换 再计算了

我们的公式中可以实现计算平方等 如Math.pow(2,2) 只要js支持的数学公式都可以写进去

至于怎么将公式中的参数全部替换 我们可以用循环遍历替换参数 这里就不写了 有了上面的方法就ok了

我有这样的想法就是因为开发中频繁编写公式 可能会出错 另一方面纯用java实现公式后期维护修改也麻烦 利用这种方法我们在后期修改公式时只需修改字符串公式即可

这样我们还可以让用户自定义公式

当然我们还可以利用ScriptEngineManager他实现执行js代码块 如if语句等 下面是个例子 可以参考下

public Map<String,Object> myFunction(String formula){
ScriptEngineManager manager = new ScriptEngineManager();
ScriptEngine engine = manager.getEngineByName("js");
Object result;
try {
engine.eval(formula);
Invocable inv = (Invocable) engine;
result=inv.invokeFunction("js");
Object function = engine.get("useFunction");
map.put("result", result);
map.put("function", function);
} catch (Exception e) {
result=Double.NaN;
}
return map;
}

以上这篇利用ScriptEngineManager实现字符串公式灵活计算的方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • java把字符串转化成公式计算的示例

    如下所示: static ScriptEngine jse = new ScriptEngineManager().getEngineByName("JavaScript"); Object aa = jse.eval(indexFormula) 以上这篇java把字符串转化成公式计算的示例就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们.

  • Java调用JavaScript实现字符串计算器代码示例

    如果表达式是字符串的形式,那么一般我们求值都会遇到很大的问题. 这里有一种直接调用JavaScript的方法来返回数值,无疑神器. 代码如下: package scc; import javax.script.ScriptEngine; import javax.script.ScriptEngineManager; import javax.script.ScriptException; public class Counter { public static void main(String

  • 利用ScriptEngineManager实现字符串公式灵活计算的方法

    在开发中我们可能会遇到好多不固定的公式计算 如有时候我们需要将excel中大量的计算公式转换成java语言进行实现 如果说单纯的用java的基本方法实现计算 我们就需要的分析excel中的公式将其一一并转换成java代码 这样对我们开发人员来说工作量有点太大了 而且在转换过程中很可能会出错 下面就介绍一种简单的实现方法 我们知道js的eval()方法可以执行字符串的代码 而恰好jdk6增加了对脚本语言的支持 我们可以利用这个特性对计算实现简单化的处理 下面举个例子 加入有个公式 A+B*C A=

  • Java计算字符串公式的方式解读

    目录 如何计算字符串公式 解决方案一 解决方案二 总结 如何计算字符串公式 解决方案一 使用 commons-jexl3 jar 包 可以使用 commons-jexl3 jar包,此 jar 包提供了一些方法用于计算字符串中的公式. maven 依赖如下: <dependency>     <groupId>org.apache.commons</groupId>     <artifactId>commons-jexl3</artifactId&g

  • 利用JS判断字符串是否含有数字与特殊字符的方法小结

    前言 本文主要介绍的是利用JS判断字符串是否含有数字与特殊字符的方法,文中有几种不同的方法,包括普通的JS验证法.正则表达式法,另外还有判断是否为浮点数的js函数,在最后还将简要介绍下isNAN函数的使用方法和例子,来一起学习学习吧. 一.正则表达式方法判断是否为数字,包括判断正整数: function checkRate(input) { var re = /^[0-9]+.?[0-9]*$/; //判断字符串是否为数字,//若判断正整数,则后边是:/^[1-9]+[0-9]*]*$/ if

  • 使用java处理字符串公式运算的方法

    在改进一个关于合同的项目时,有个需求,就是由于合同中非数据项的计算公式会根据年份而进行变更,而之前是将公式硬编码到系统中的,只要时间一变,系统就没法使用了,因此要求合同中各个非基础数据的项都能自定义公式,根据设置的公式来自动生成报表和合同中的数据. 显然定义的公式都是以字符串来存储到数据库的,可是java中没有这种执行字符串公式的工具或者类,而且是公式可以嵌套一个中间公式.比如:基础数据dddd是56,而一个公式是依赖dddd的,eeee=dddd*20,而最终的公式可能是这样:eeee*-12

  • java中利用栈实现字符串回文算法

    问题 给定一个由多个a和b组成的字符串数组,字符串中有一个特殊的字符X,位于字符串的正中间,例如(aaaabbbbXabaabbbb),如何判定该字符串是否回文 简单算法 定义两个下标分别指向字符串的头和尾,每次比较两个下标位置的值是否相等,如果不相等,那么输入的 字符串不是回文,如果相等,左边的下表加1,右边的下表减1,重复上述步骤直至两个下标都指向字符串的正中间或者确定字符串不是回文 /** * 判断字符串是否是回文 */ public int isPalindrome(String inp

  • 如何利用JavaScript获取字符串中重复次数最多的字符

    目录 题目 分析 使用对象 解题思路: 代码实现如下: 分析: 数组&指针 解题思路: 代码实现如下: 分析: 总结 想要保持自己的技术活力,最有效的手段就是通过不断地输入来提供足够的养分.我们也不必刻意追求高深的或者新鲜的知识点,通过对一个基础问题的全方位多维度解析,同样也会收获不小. 题目 假设有这么一道题目:请获取字符串 "bianchengsanmei,xuexiyouqudezhishi,jieshiyouqudepengyou,suzaoyouqudelinghun.&quo

  • 利用Python读取CSV文件并计算某一列的均值和方差

    近日需要对excel的csv文件进行处理,求取某银行历年股价的均值方差等一系列数据 文件的构成很简单,部分如下所示 总共有接近七千行数据,主要的工作就是将其中的股价数据提取出来,放入一个数组之中,然后利用numpy模块即可求出需要的数据. 这里利用了csv模块来对文件进行处理,最终实现的代码如下: import csv import numpy as np with open('pingan_stock.csv') as csv_file: row = csv.reader(csv_file,

  • C语言利用sprintf固定字符串输出位数

    目录 一.前言 二.案例 [1]多个整数输出补齐 [2]单个整数输出补齐 [3]固定时间长度 [4]浮点数补齐 [5]onenet数据组合 一.前言 sprintf 函数是一个 C 语言中的函数,也被许多其他编程语言所支持.它允许你根据一组变量来格式化和存储一个字符字符串.sprintf 函数可以用于将格式化的数据写入字符数组或字符串中,然后根据需要打印或存储该字符串.sprintf 函数的第一个参数是格式化数据将被写入的字符数组或字符串,随后的参数是用于生成格式化输出的变量.sprintf 函

  • SQL Server中利用正则表达式替换字符串的方法

    建立正则替换函数,利用了OLE对象,以下是函数代码: --如果存在则删除原有函数 IF OBJECT_ID(N'dbo.RegexReplace') IS NOT NULL DROP FUNCTION dbo.RegexReplace GO --开始创建正则替换函数 CREATE FUNCTION dbo.RegexReplace ( @string VARCHAR(MAX), --被替换的字符串 @pattern VARCHAR(255), --替换模板 @replacestr VARCHAR

  • Shell脚本中计算字符串长度的5种方法

    有时在Linux操作系统中需要计算某个字符串的长度,通过查询资料整理了下目前Shell中获取字符串的长度的多种方法,在这里分享给大家,方法如下: 方法1: 使用wc -L命令 wc -L可以获取到当前行的长度,因此对于单独行的字符串可以用这个简单的方法获取,另外wc -l则是获取当前字符串内容的行数. 复制代码 代码如下: echo "abc" |wc -L 方法2: expr length string 使用expr length可以获取string的长度 方法3: awk获取域的个

随机推荐