将数学字符串转换为 int

Posted

技术标签:

【中文标题】将数学字符串转换为 int【英文标题】:Convert mathematical string to int 【发布时间】:2011-03-16 05:42:52 【问题描述】:

有没有一种简单的方法来获取诸如“5*4”之类的字符串并返回 20?

【问题讨论】:

这个问题复制到***.com/questions/2607798/… @Adeel:不,不是... C# 和 Java 是具有不同库和程序的不同语言。 @Adeel - 绝对不是,另一个问题要求 c# 解决方案。 【参考方案1】:

最简单的方法是使用 JRE 6 标准 API 提供的 Rhino javascript engine。

编辑: 根据 cmets,如果字符串是用户提供的,这可能是一个潜在的安全漏洞。请务必过滤掉除数字、大括号和数学运算符之外的所有内容。

【讨论】:

是的……但是除了 JavaScript 引擎之外,它还能做更多的事情。如果您在某些情况下暴露了 Rhino 评估程序,您可能会引入……咳咳……安全问题。 这种事情总是让我想起SQL代码注入漏洞。如果这些字符串来自程序外部(用户、文件、套接字......),为了确保它们不是潜在的安全风险,您可能需要解析它们。好吧,好吧,不完全是-您可能会通过正则表达式检查而侥幸,因为您只关心合法令牌集。并且 Javascript 没有标准的文件操作等,所以无论如何你可能是安全的。仍然 - 这让我很紧张。 嗯——我真的应该在评论之前刷新一下。哦,好吧。【参考方案2】:

我不知道哪些是最好的,但有“数学表达式评估器”包。

查看Java Math Expression Evaluator(包括one file源代码)

网站使用示例:

java -cp meval.jar com.primalworld.math.MathEvaluator -cos(0)*(1+2) java -cp meval.jar com.primalworld.math.MathEvaluator .05*200+3.01

【讨论】:

因为我正在为第 5 版开发,所以这个最适合我。【参考方案3】:

您可能正在寻找类似JbcParser - Math Parser for Java 的东西。

【讨论】:

虽然也许值得指出它不是免费的(在任何一种意义上)。 认真的阿尔乔姆? “This”是一个可怕的链接描述符,回滚我的编辑是完全无知的。【参考方案4】:

我用谷歌搜索了一下,发现了一个here。它不完全是您所需要的,但也许它会有所帮助。

import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;

public class ScriptDemo 

    public static void main(String[] args) 

        ScriptEngineManager manager = new ScriptEngineManager();
        ScriptEngine engine = manager.getEngineByName("js");

        engine.put("a", 1);
        engine.put("b", 2);

        try 
            String expression = "(a + b) > 2";
            Object result = engine.eval(expression);
            System.out.println(expression+" ? "+result);
         catch(ScriptException se) 
            se.printStackTrace();
        
    

【讨论】:

【参考方案5】:

有关表达式评估的更多详细信息,请参见:

Algorithms in Java, Volume 1, Parts 1-4

【讨论】:

【参考方案6】:

听起来你应该看看JEXL(Java 表达式语言)

非常容易使用;例如,您的问题的解决方案是:

  public static void main(String[] args) 
    long a = 5;
    long b = 4;
    String theExpression = "a * b";
    JexlEngine jexl = new JexlEngine();
    Expression e = jexl.createExpression(theExpression);
    JexlContext context = new MapContext();
    context.set("a", a);
    context.set("b", b);
    Long result = (Long) e.evaluate(context);
    System.out.println("The answer : " + result);
  

如果直接读入字符串,您也可以使用以下内容:

 public static void main(String[] args) 
   JexlEngine jexl = new JexlEngine();
   Expression e = jexl.createExpression("5 * 4");
   Integer result = (Integer) e.evaluate(null);
   System.out.println("The answer : " + result);
 

【讨论】:

【参考方案7】:

您最好为此使用库。 Janino 能够评估任意 Java 表达式,例如您指定的表达式,等等。

查看ExpressionEvaluator 示例。

【讨论】:

【参考方案8】:

您可以尝试使用 Integer.parseInt() 自己解析它并使用 switch() 语句来查找运算符。

您也可以尝试使用 javax.script.ScriptEngine;如需更多信息,请参阅http://forums.sun.com/thread.jspa?threadID=5144807。

【讨论】:

如果所有的表达式都是二数一运算符,也许。在现实世界中,你对“5+4*3”有什么期望——我希望是 17 而不是 27,这意味着优先级很重要。优先级解析并不难,但它肯定不是微不足道的。我同意“使用图书馆”的答案。【参考方案9】:

您也可以使用BeanShell。 它实际上更像是一个 Java 源代码解释器,但最近我用它来评估一些包含变量的表达式(只有 eval 方法使用的是 BeanShell,rest 用于准备输出):

import bsh.EvalError;
import bsh.Interpreter;

public class EVAL 

    private static final String FORMAT = "%-5s | %-5s | %-5s | %s%n";

    public static void main(String[] args) 
        tabela("((a && b)||c)");
        tabela("a ? (b || c) : (b && c)");
        tabela("(a?1:0) + (b?1:0) + (c?1:0) >= 2");
    

    private static void tabela(String expressao) 
        System.out.printf(FORMAT, "  a  ", "  b  ", "  c  ", expressao);
        System.out.printf(FORMAT, "-----", "-----", "-----", expressao.replaceAll(".", "-"));
        try 
            for (int i = 0; i < 8; i++) 
                boolean a = (i & (1<<2)) != 0;
                boolean b = (i & (1<<1)) != 0;
                boolean c = (i & (1<<0)) != 0;
                boolean r = eval(expressao, a, b, c);
                System.out.printf(FORMAT, a, b, c, r);
            
         catch (EvalError ex) 
            ex.printStackTrace();
        
        System.out.println();
        System.out.println();
    

    private static boolean eval(String expressao, boolean a, boolean b, boolean c) throws EvalError 
        Interpreter inter = new Interpreter();
        inter.set("a", a);
        inter.set("b", b);
        inter.set("c", c);
        Object resultado = inter.eval(expressao);
        return (Boolean) resultado;
    

结果:

  a   |   b   |   c   | ((a && b)||c)
----- | ----- | ----- | -------------
false | false | false | false
false | false | true  | true
false | true  | false | false
false | true  | true  | true
true  | false | false | false
true  | false | true  | true
true  | true  | false | true
true  | true  | true  | true


  a   |   b   |   c   | a ? (b || c) : (b && c)
----- | ----- | ----- | -----------------------
false | false | false | false
false | false | true  | false
false | true  | false | false
false | true  | true  | true
true  | false | false | false
true  | false | true  | true
true  | true  | false | true
true  | true  | true  | true


  a   |   b   |   c   | (a?1:0) + (b?1:0) + (c?1:0) >= 2
----- | ----- | ----- | --------------------------------
false | false | false | false
false | false | true  | false
false | true  | false | false
false | true  | true  | true
true  | false | false | false
true  | false | true  | true
true  | true  | false | true
true  | true  | true  | true

【讨论】:

【参考方案10】:

我过去使用过JEval,发现它非常简单直观。这是一个代码sn-p:

import net.sourceforge.jeval.EvaluationException;
import net.sourceforge.jeval.Evaluator;

public class Main 
    public static void main(String[] args) 
        try 
            System.out.println(new Evaluator().evaluate("5+4*3"));
        
        catch (EvaluationException e) 
            e.printStackTrace();
        
    

【讨论】:

【参考方案11】:

还有这个:

https://eval.dev.java.net/

【讨论】:

以上是关于将数学字符串转换为 int的主要内容,如果未能解决你的问题,请参考以下文章

在C语言中,输入一个数值,将它转换成弧度,怎么转换?

C语言如何将int类型转换成字符串

如何将切片字符串中的单个字符转换为 Python 中的整数?

怎么将字符串类型转换成int型?初学者简单方法

怎么将字符串类型转换成int型?初学者简单方法

将字符串转换为 int 和 char