Java表达式求值引擎Aviator(一)

2021/4/18 20:28:06

本文主要是介绍Java表达式求值引擎Aviator(一),对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

环境:Java8 + Aviator5.2.5


Aviator简介

Aviator 是一个高性能、轻量级的 java 语言实现的表达式求值引擎,主要用于各种表达式的动态求值。现在已经有很多开源可用的 java 表达式求值引擎,为什么还需要 Avaitor 呢?
Aviator 的设计目标是轻量级和高性能 ,相比于 Groovy、JRuby 的笨重,Aviator 非常小,加上依赖包也才 450K,不算依赖包的话只有 70K;当然,Aviator 的语法是受限的,它不是一门完整的语言,而只是语言的一小部分集合。
其次,Aviator 的实现思路与其他轻量级的求值器很不相同,其他求值器一般都是通过解释的方式运行,而 Aviator 则是直接将表达式编译成 Java 字节码,交给 JVM 去执行。简单来说,Aviator 的定位是介于 Groovy 这样的重量级脚本语言和 IKExpression 这样的轻量级表达式引擎之间。

Aviator特性

  • 支持大部分运算操作符,包括算术操作符、关系运算符、逻辑操作符、位运算符、正则匹配操作符(=~)、三元表达式?: ,并且支持操作符的优先级和括号强制优先级,具体请看后面的操作符列表。
  • 支持函数调用和自定义函数
  • 内置支持正则表达式匹配,类似 Ruby、Perl 的匹配语法,并且支持类 Ruby 的$digit指向匹配分组。
  • 自动类型转换,当执行操作的时候,会自动判断操作数类型并做相应转换,无法转换即抛异常。
  • 支持传入变量,支持类似 a.b.c 的嵌套变量访问。
  • 函数式风格的 seq 库,操作集合和数组。

Aviator 的限制:

  • 没有 if else、do while 等语句,没有赋值语句,仅支持逻辑表达式、算术表达式、三元表达式和正则匹配。
  • 不支持八进制数字字面量,仅支持十进制和十六进制数字字面量。

整体结构

Aviator 的结构非常简单,一个典型的求值器的结构。

Java表达式求值引擎Aviator(一)

 


使用示例

依赖

<dependency>
  <groupId>com.googlecode.aviator</groupId>
  <artifactId>aviator</artifactId>
  <version>5.2.5</version>
</dependency>
<dependency>
  <groupId>commons-beanutils</groupId>
  <artifactId>commons-beanutils</artifactId>
  <version>1.9.4</version>
</dependency>
<dependency>
  <groupId>commons-logging</groupId>
  <artifactId>commons-logging</artifactId>
  <version>1.2</version>
</dependency>

示例1:

执行简单的加法运算

public class SimpleExample {
  public static void main(String[] args) {
    Long result = (Long) AviatorEvaluator.execute("1+2+3");
    System.out.println(result);
  }
}

结果是 Long,而不是 Integer。这是因为 Aviator 的数值类型仅支持Long 和 Double,任何整数都将转换成 Long,任何浮点数都将转换为 Double,包括用户传入的变量数值。

示例2:

给表达式传递参数

public class InputParamsExample {
	
  public static void main(String[] args) {
    Map<String, Object> env = new HashMap<String, Object>();
    env.put("name", "张三");
    String result = (String) AviatorEvaluator.execute(" '你的姓名是' + name ", env) ;
    System.out.println(result) ;
  }
	
}

输出:你的姓名是张三。

示例3:

exec 方法

Aviator 2.2 开始新增加一个 exec 方法,可以更方便地传入变量并执行,而不需要构造 env 这个 map 了:

public class ExecParamsExample {
	
  public static void main(String[] args) {
    String myname="dennis";
    Object result = AviatorEvaluator.exec(" 'hello ' + name ", myname);
    System.out.println(result) ;
  }
	
}

示例4:

调用函数

Aviator 支持函数调用,函数调用的风格类似Lua脚本语法。

public class InvokeFunctionExample {
	
  public static void main(String[] args) {
    Object res1 = AviatorEvaluator.execute("string.length('hello')");
    Object res2 = AviatorEvaluator.execute("string.substring('hello',1,2)") ;
    Object res3 = AviatorEvaluator.execute("string.contains('hello','ll')") ;
    Object res4 = AviatorEvaluator.exec("string.length(name)", "我是中国人") ;
    System.out.println(res1 + "\t" + res2 + "\t" + res3 + "\t" + res4) ;
  }
	
}

示例5:

自定义函数

Aviator 除了内置的函数之外,还允许用户自定义函数,只要实现
com.googlecode.aviator.runtime.type.AviatorFunction 接口,并注册到 AviatorEvaluator 即可使用

AviatorFunction 接口十分庞大,通常来说你并不需要实现所有的方法,只要根据你的方法的参数个数,继承 AbstractFunction 类并 override 相应方法即可。

自定义函数:

public class AddFunction extends AbstractFunction {
	
  @Override
  public AviatorObject call(Map<String, Object> env, AviatorObject arg1, AviatorObject arg2) {
    Number left = FunctionUtils.getNumberValue(arg1, env);
    Number right = FunctionUtils.getNumberValue(arg2, env);
    return new AviatorDouble(left.doubleValue() + right.doubleValue());
  }
  // 定义(返回)函数的名称。
  public String getName() {
    return "add";
  }
	
}

注册函数:

public class CustomFunctionExample {
	
  public static void main(String[] args) {
    //注册函数
    AviatorEvaluator.addFunction(new AddFunction());
    System.out.println(AviatorEvaluator.execute("add(10,100)"));
    // 删除函数通过:removeFunction
  }
	
}

示例6:

编译表达式

我们可以自己先编译表达式,返回一个编译的结果,然后传入不同的 env 来复用编译结果,提高性能,这是更推荐的使用方式。

public class CompileExample {
  public static void main(String[] args) {
    String expression = "a-(b-c) > 100";
    // 编译表达式
    Expression compiledExp = AviatorEvaluator.compile(expression);
    Map<String, Object> env = new HashMap<String, Object>();
    env.put("a", 100.3d);
    env.put("b", 45);
    env.put("c", -199.100d);
    // 执行表达式
    Boolean result = (Boolean) compiledExp.execute(env);
    System.out.println(result);
  }
}

通过 compile 方法可以将表达式编译成 Expression 的中间对象,当要执行表达式的时候传入env 并调用 Expression 的 execute 方法即可。表达式中使用了括号来强制优先级,这个例子还使用了>用于比较数值大小,比较运算符!=、==、>、>=、<、<=不仅可以用于数值,也可以用于 String、Pattern、Boolean 等等,甚至是任何用户传入的两个都实现了 java.lang.Comparable 接口的对象之间。

示例7:

访问数组和集合

可以通过中括号去访问数组和 java.util.List 对象,可以通过 map.key 访问 java.util.Map 中 key对应的 value。

public class CollectionExample {
  public static void main(String[] args) {
    List<String> list = new ArrayList<>();
    list.add("我是");
    list.add("中国人");
    int[] array = new int[3];
    array[0] = 10;
    array[1] = 100;
    array[2] = 1000;
    final Map<String, Object> map = new HashMap<>();
    map.put("date", LocalDate.now());
    Map<String, Object> env = new HashMap<>();
    env.put("list", list);
    env.put("array", array);
    env.put("mmap", map);
    System.out.println(AviatorEvaluator.execute(
				"list[0]+list[1]+'\narray[0]+array[1]+array[2]='+(array[0]+array[1]+array[2]) +' \ntoday is '+ mmap.date ",
				env));
  }
}

下一篇继续结束:三元操作符,正则表达式匹配,日期比较,大数计算和精度等等。。

给个关注+转发呗谢谢

Java表达式求值引擎Aviator(一)

 

Java表达式求值引擎Aviator(一)

 

Java表达式求值引擎Aviator(一)

 

Java表达式求值引擎Aviator(一)



这篇关于Java表达式求值引擎Aviator(一)的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!


扫一扫关注最新编程教程