`
shirui_java
  • 浏览: 10698 次
  • 性别: Icon_minigender_1
  • 来自: 郑州
文章分类
社区版块
存档分类
最新评论

java 字符串公式计算 --------转载

阅读更多
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* 实现对字符串表达式进行简单的包括函数的运算。
*  @author 时锐
*/
public class AnalyzeCalculate {
/**
* 定义运算符
*/
public static List<String> lc = new ArrayList<String>();
static {
lc.add("+");
lc.add("`");
lc.add("*");
lc.add("/");
}
/**
* 定义逻辑运算符
*/
public static List<String> lj = new ArrayList<String>();
static {
lj.add(">");
lj.add("<");
lj.add("=");
lj.add("!");
}
/**
* int数组注释,第一个标识:0自带函数 1自定义函数;第二个标识:参数个数
*/
public static Map<String, int[]> funMap = new HashMap<String, int[]>();
static {
// 自带函数,可利用反射机制
funMap.put("abs", new int[] { 0, 1 });
funMap.put("acos", new int[] { 0, 1 });
funMap.put("asin", new int[] { 0, 1 });
funMap.put("atan", new int[] { 0, 1 });
funMap.put("cbrt", new int[] { 0, 1 });
funMap.put("ceil", new int[] { 0, 1 });
funMap.put("cos", new int[] { 0, 1 });
funMap.put("cosh", new int[] { 0, 1 });
funMap.put("exp", new int[] { 0, 1 });
funMap.put("expm1", new int[] { 0, 1 });
funMap.put("floor", new int[] { 0, 1 });
funMap.put("log", new int[] { 0, 1 });
funMap.put("log10", new int[] { 0, 1 });
funMap.put("log1p", new int[] { 0, 1 });
funMap.put("random", new int[] { 0, 1 });
funMap.put("rint", new int[] { 0, 1 });
funMap.put("round", new int[] { 0, 1 });
funMap.put("signum", new int[] { 0, 1 });
funMap.put("sin", new int[] { 0, 1 });
funMap.put("sinh", new int[] { 0, 1 });
funMap.put("sqrt", new int[] { 0, 1 });
funMap.put("tan", new int[] { 0, 1 });
funMap.put("tanh", new int[] { 0, 1 });
funMap.put("max", new int[] { 0, 2 });
funMap.put("min", new int[] { 0, 2 });
// 自定义函数
funMap.put("if", new int[] { 1, 3 });
funMap.put("mod", new int[] { 1, 2 });
funMap.put("int", new int[] { 1, 1 });
}

/**
* 公式初始化转换
*
* @param str
* @return 处理过的计算客串
*/
public static String strCast(String str) {
str = str.toLowerCase();// 去除空格,变小写

if (str == null ? true : str.length() == 0)
return "0";
if (!checkFormula(str))
return "0";
str = str.replaceAll("\\*-", "**");
str = str.replaceAll("-\\*", "**");
str = str.replaceAll("/-", "//");
str = str.replaceAll("-/", "//");
str = str.replaceAll("\\+-", "-");
str = str.replaceAll("-\\+", "-");
str = str.replaceAll("-", "`");
str = str.replaceAll("\\*\\*", "*-");
str = str.replaceAll("//", "/-");
str = str.replaceAll(" ", "");
return str;
}

/**
* 检查公式中括号出现次数是否正确
*
* @param formulaStr
* @return 公式中的括号是否成对
*/
public static boolean checkFormula(String formulaStr) {
boolean flag = true;
int count = 0;
for (int i = 0; i < formulaStr.length(); i++) {
String s = String.valueOf(formulaStr.charAt(i));
if ("(".equals(s))
count++;
else if (")".equals(s))
count--;
}
flag = count == 0;
return flag;
}

/**
* 分割函数
*
* @param str
* @param bs
* @return 分割后的客串
*/
public static String[] spliteFun(String str, String bs) {
List<String> list = new ArrayList<String>();
String bds = "";
int bracket = 0;
int len = str.length();
for (int i = 0; i < len; i++) {
String s = String.valueOf(str.charAt(i));
if ("(".equals(s)) {
bracket++;
} else if (")".equals(s)) {
bracket--;
}

if (bracket == 0 && bs.equals(s)) {
list.add(bds);
bds = "";
continue;
}

bds += s;
}

list.add(bds);

String[] ss = new String[list.size()];
for (int i = 0; i < list.size(); i++) {
ss[i] = list.get(i);
}

return ss;
}

/**
* 用户自定义函数
*
* @param str
* @param funStr
* @return 处理用户自定义函数
*/
public static String customFun(String str, String funStr) {
String reval = "0";

String[] gss = spliteFun(str, ",");
if ("if".equals(funStr)) {
if (compare(gss[0])) {
reval = calculate(gss[1]);
} else {
reval = calculate(gss[2]);
}
} else if ("mod".equals(funStr)) {
double d2 = new Double(calculate(gss[1]));
if (d2 == 0)
return reval;
double d1 = new Double(calculate(gss[0]));
reval = (d1 % d2) + "";
} else if ("int".equals(funStr)) {
reval = Math.floor(new Double(calculate(gss[0]))) + "";
}
return reval;
}

/**
* 逻辑表达式判断
* @param str
* @return true or false
*/
public static boolean compare(String str) {
boolean flag = false;
boolean bs = false;
int len = str.length();
int bracket = 0;
String ljbds = "";
double d_left = 0;
double d_right = 0;

for (int i = 0; i < len; i++) {
String s = String.valueOf(str.charAt(i));
if ("(".equals(s)) {
bracket++;
} else if (")".equals(s)) {
bracket--;
}

if (bracket == 0 && lj.contains(s)) {
for (int j = i; j < len; j++) {
String ts = String.valueOf(str.charAt(j));
if (lj.contains(ts)) {
ljbds += ts;
} else {
bs = true;
break;
}
}
}
if (bs)
break;
}

String[] s = str.split(ljbds);
d_left = new Double(calculate(s[0]));
d_right = new Double(calculate(s[1]));

if ("<".equals(ljbds)) {
if (d_left < d_right)
return true;
} else if (">".equals(ljbds)) {
if (d_left > d_right)
return true;
} else if ("=".equals(ljbds)) {
if (d_left == d_right)
return true;
} else if (">=".equals(ljbds)) {
if (d_left >= d_right)
return true;
} else if ("<=".equals(ljbds)) {
if (d_left <= d_right)
return true;
} else if ("<>".equals(ljbds) || "!=".equals(ljbds)) {
if (d_left != d_right)
return true;
}

return flag;
}

/**
* 递归调用运算
*
* @param str
* @return String
*/
public static String calculate(String str) {

String reval = "";
String bds = "";
int bracket = 0;// 对应括号个数
int pos = 0;
boolean title = false;

if (str.substring(0, 1).equals("`")) {
str = str.substring(1);
title = true;
}

int len = str.length();

for (int i = 0; i < len; i++) {
String s = String.valueOf(str.charAt(i));
pos = i;
bracket = 0;
if (!lc.contains(s)) {// 如果没遇到运算符
if ("(".equals(s)) {// 如果遇到左括号
if (funMap.containsKey(bds)) {// 如果左括号前是函数
for (int j = i + 1; j < len; j++) {// 从左括号后开始循环
pos++;// 累计移动字符位数
String ts = String.valueOf(str.charAt(j));// 单个字符
// reval+=ts;
if ("(".equals(ts))// 如果是左括号累计
bracket++;
else if (")".equals(ts)) {// 如果是右括号进行减少
bracket--;
if (bracket == -1) {// 如果是-1,标识括号结束
reval = reval.substring(0, reval.length()
- bds.length());// 重新获得去掉函数头的表达式
reval += funCalculate(str.substring(i + 1,
j), bds);// 表达式加上函数结果,形成新表达式
i = pos;// 计数器增加
bds = "";// 函数头清空
break;// 退出本次循环
}
}
}
} else {// 如果是普通运算
for (int j = i + 1; j < len; j++) {
pos++;
String ts = String.valueOf(str.charAt(j));
if ("(".equals(ts))
bracket++;
else if (")".equals(ts)) {
bracket--;
if (bracket == -1) {
reval += calculate(str
.substring(i + 1, pos));
i = pos;
bds = "";
break;
}
}
}
}
} else {// 累加总表达式和最后一个运算数(或函数)
bds += s;
reval += s;
}
} else {// 遇到运算符最后一个运算数(或函数)清空
bds = "";
reval += s;
}
}

if (title)
reval = "-" + reval;
return szys(reval);
}

/**
* 函数运算
*
* @param gs
* @param flag
* @return String
*/
public static String funCalculate(String gs, String funStr) {
String rval = "0";
if (funMap.containsKey(funStr)) {
int[] csi = funMap.get(funStr);
try {
if (csi[0] == 0) {// java内部函数,通过反射调用
Class[] cs = new Class[csi[1]];
Object[] objs = new Object[csi[1]];
String[] gss = zlcs(gs);
for (int i = 0; i < csi[1]; i++) {
cs[i] = double.class;
objs[i] = new Double(calculate(gss[i]));
}
Class cls = Class.forName("java.lang.Math");
Method m = cls.getMethod("abs", cs);
rval = String.valueOf(m.invoke(cls, objs));
} else if (csi[0] == 1) {// 自定义函数
rval = customFun(gs, funStr);
}
} catch (Exception e) {

}
}

return rval;
}

/**
* 公式里的参数分割
* @param str
* @return String[]
*/
public static String[] zlcs(String str) {
int len = str.length();
boolean flag = true;
String tstr = "";

for (int i = 0; i < len; i++) {
String s = String.valueOf(str.charAt(i));
if ("(".equals(s)) {
flag = false;
} else if (")".equals(s)) {
flag = true;
}
if (flag && ",".equals(s)) {
tstr += "@";
} else {
tstr += s;
}
}

return tstr.split("@");

}

/**
* 四则运算表达式处理
*
* @param str
* @return String
*/
public static String szys(String gs) {
gs = gs + "+0"; // 因为下面的计算是遇到符号才进行,所以多加入一个计算符号,不影响值.
String c1 = "";// 第一个运算符号
String c2 = "";// 第二个运算符号
String s1 = "";// 第一个运算数
String s2 = "";// 第二个运算数
String s3 = "";// 第三个运算数

int len = gs.length();
for (int i = 0; i < len; i++) {
String s = String.valueOf(gs.charAt(i));// 获得该位置字符并转换成字符串做比较

if (lc.contains(s)) { // 如果是运算符号
if (c1.length() == 0)// 如果第一个运算符号为空,加入
c1 = s;
else if (c2.length() == 0) {// 否则,如果第二个运算符号为空,加入
c2 = s;// 第二个运算符号
if ("+".equals(c2) || "`".equals(c2)) {// 如果第二个运算符号级别低,那么进行计算
s1 = _4zys(s1, c1, s2);// 第一个和第二个数计算
c1 = c2;// 保存第二个运算符,其他为空
c2 = "";
s2 = "";
}
} else {// 上述都保存过
if ("+".equals(s) || "`".equals(s)) {// 如果第三个运算符级别低,进行运算
s2 = _4zys(s2, c2, s3);// 先算第二三个数,保存至第二个
s1 = _4zys(s1, c1, s2);// 再算第一二个,保存至第一个
c1 = s;// 保存当前运算符,其他为空
s2 = "";
c2 = "";
s3 = "";
} else {// 如果第三个运算符级别高
s2 = _4zys(s2, c2, s3);// 先算第二三个数,保存至第二个
c2 = s;// 前面不动,保存运算符
s3 = "";
}
}
} else if (s1.length() > 0 && c1.length() > 0 && c2.length() == 0) {// 如果第一个数,第一个运算符已保存,第二个运算符未保存,保存第二哥数
s2 += s;
} else if (c1.length() == 0) {// 如果没有运算符,保存第一个数
s1 += s;
} else if (s1.length() > 0 && s2.length() > 0 && c1.length() > 0
&& c2.length() > 0) {// 如果第一二个数和运算符都有,保存第三个数
s3 += s;
}
}
return s1;
}

/**
* 基本四则运算
*
* @param c1
*            运算数1
* @param s1
*            运算符(加减乘除)
* @param c2
*            运算数2
* @return String
*/
public static String _4zys(String c1, String s1, String c2) {
String reval = "0";

try {
double ln = Double.valueOf(c1).doubleValue();
double rn = Double.valueOf(c2).doubleValue();
if ("+".equals(s1)) {
return (ln + rn) + "";
} else if ("`".equals(s1)) {
return (ln - rn) + "";
} else if ("*".equals(s1)) {
return (ln * rn) + "";
} else if ("/".equals(s1)) {
if (rn == 0)
return reval;
else
return (ln / rn) + "";
}
} catch (Exception e) {
} finally {
}

return reval;
}

/**
* 测试
* @param args
*/
public static void main(String[] args) {

String gs = "if(4 > 3,if( ( 2 - 1 ) /  ( 0.0001 ) * 3 > 3  , ( 2 - ( 2 - 1 ) /  ( 0.0001 ) * 3  ) * 0.8 ,0),  ( 2 + ( 3 - 2 ) /  ( 0.0001 ) *1  )  * 1)";
// 进行四则运算获得结果
System.out.println(szys(strCast("3-4*-5+5")));
System.out.println(calculate(strCast("3-4*-5+5")));
// 函数运算结果
System.out
.println(calculate(strCast("3+abs(mod(4,if(2>3,1,3))-abs(2*4))*5+5")));

System.out.println(calculate(strCast("if(0.0*2>=10.0,0,10.0-0.0*2)")));

System.out.println(calculate(strCast("if(0<2,(10.0-0*5)*0.8,0)")));

System.out.println(calculate(strCast("-10/0")));

}

}

分享到:
评论
2 楼 278506470 2009-09-09  
楼主有一个地方有一点问题,利用Math类调用的是由,楼主写死了,只能进行abs形式的计算。
1 楼 tnsmat163dotcom 2009-01-20  
谢谢共享,请教:if的逻辑是什么?

相关推荐

    Java 所有字符串转UTF-8 万能工具类-GetEncode.java

    不需要关心接受的字符串编码是UTF_8还是GBK,还是ios-8859-1,自动转换为utf-8编码格式,无需判断字符串原有编码,用法://处理编码String newStr = GetEncode.transcode(oldStr);

    字符串操作-----

    字符串操作----- 字符串操作----- 字符串操作----- 字符串操作----- v 字符串操作----- 字符串操作----- 字符串操作-----

    C#中字符串转换为计算公式-并进行计算的方法(自定义公式的计算)

    ### C#中字符串转换为计算公式并进行计算的方法(自定义公式的计算) 在软件开发过程中,有时我们需要处理用户输入的动态表达式或公式,并能够即时计算这些表达式的值。这种需求常见于科学计算器应用、数学软件或者...

    java 字符串 正则表达式 详细实例代码

    在Java编程语言中,字符串(String)是至关重要的数据类型,用于存储和操作文本。字符串类提供了丰富的API,使得处理字符串变得高效且灵活。本篇将深入探讨Java中的字符串、正则表达式及其在实际编程中的详细实例代码...

    7-65 字符串替换--字符串.c

    本题目来自PTA习题集,本人在同时学习的情况下,分享自己的练习情况!

    java字符串的各种编码转换

    根据提供的文件信息,本文将详细解释Java中字符串的不同编码转换方法及原理,并深入探讨每种编码格式的特点。 ### Java字符串的编码转换 在Java中,处理不同字符集之间的字符串转换是一项常见任务。尤其是在处理...

    java各种数据库连接字符串(2-2).jsp

    详细讲解java各种数据库连接字符串(2-2).jsp

    Java 字符串常用方法

    以下是一些Java `String`类中常用的方法,这些方法对于理解和操作字符串至关重要。 1. **创建字符串** - `new String()`: 使用此构造函数创建一个新的字符串对象,可以传入字符数组或另一个字符串作为参数。 - `...

    java 中文字符串,utf-8编码为byte数组的计算过程

    本文将深入探讨如何将中文字符串转换成UTF-8编码的字节数组,并解析这一过程中的计算步骤。UTF-8是一种广泛使用的Unicode字符编码,它可以表示Unicode字符集中的所有字符。 首先,我们需要明白Unicode字符集是一个...

    公式解析类(字符串公式,自定义函数公式)

    ### 公式解析类(字符串公式,自定义函数公式) #### 概述 本文将详细介绍一个公式解析类,该类能够处理字符串公式以及自定义函数公式。此类在数学计算、数据分析等领域有着广泛的应用,比如在电子表格软件或编程语言...

    116、1298:计算字符串距离--2020.04.09a.pdf

    根据给定的信息,本文将对计算字符串距离这一知识点进行详细阐述。主要涉及的是字符串编辑距离(也称为Levenshtein距离)的相关算法实现。 ### 字符串编辑距离简介 字符串编辑距离是指两个字符串之间,由一个转换...

    JAVA字符串处理函数列表一览

    ### JAVA字符串处理函数列表一览 在Java编程语言中,字符串处理是极其常见且重要的操作之一。字符串类`String`提供了丰富的内置方法来帮助开发者高效地完成各种字符串操作任务。本文将详细解读`String`类中的一些...

    Java 工具类, 计算字符串表达式, 支持 +,-,*,/,%. ()

    自己写的一个计算字符串表达式工具类; Java 工具类, 计算字符串表达式, 支持 +,-,*,/,%. ()

    字符串逆序编码-java

    字符串逆序-java

    JAVA日期与字符串的转换

    在Java编程中,经常需要将日期对象转化为字符串形式以便于存储或显示,或者反过来将字符串转化成日期对象来进行日期计算等操作。本文将详细介绍如何在Java中实现这两种转换。 #### 一、将一个Date类型的变量转换为...

    工具-Java-字符串模板插值-示例

    在web开发中,字符串插值是最常用的字符串操作之一。 虽然许多编程语言都提供内置的字符串插值支持,但在这个挑战中,您需要自己实现它。不允许在您选择的编程语言中使用内置的字符串插值机制。 在这个挑战中,...

    java字符串-用户名和密码-加密解密

    本示例通过Java实现了字符串的加密与解密功能,采用DES加密算法,提供了基本的加密解密功能及字符串与十六进制之间的转换。这种实现方式适用于简单的加密需求,但对于更复杂的安全场景,建议考虑使用更先进的加密...

    VC++程序: 将字符串转换成公式并计算结果

    int Calculate(string formula) 函数功能:输入一个字符串公式string formula,允许四则混合运算,然后输入公式中对应变量的取值,最后返回int类型计算结果。 变量的格式:必须由1位字母加1位数字组成,如a1、b2等...

    java字符串数学表达式(含括号)计算值

    代码均为自己设计所写,分享一下。 字符串数学表达式(含括号)计算值 如: "31+3*3-20/2*5+40/8+4*5" ((2*(19-13*(1+2)/39)/6+4)-5)/5+((2+3)*2-5)

    第4节: 揭秘JVM字符串常量池和Java堆-01

    第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: ...

Global site tag (gtag.js) - Google Analytics