“初步了解和使用Aviator”
1.Aviator简介
Aviator 是一个高性能,轻量级的java语言实现的表达式求值引擎,主要用于各种表达式的动态求值。
官方文档
github地址
支持数字、字符串、正则表达式、布尔值、正则表达式等基本类型,完整支持所有 Java 运算符及优先级等。
函数是一等公民,支持闭包和函数式编程。
内置 bigint/decmal 类型用于大整数和高精度运算,支持运算符重载得以让这些类型使用普通的算术运算符 ±*/ 参与运算。
完整的脚本语法支持,包括多行数据、条件语句、循环语句、词法作用域和异常处理等。
函数式编程结合 Sequence 抽象,便捷处理任何集合。
轻量化的模块系统。
多种方式,方便地调用 Java 方法,完整支持 Java 脚本 API(方便从 Java 调用脚本)。
丰富的定制选项,可作为安全的语言沙箱和全功能语言使用。
轻量化,高性能,通过直接将脚本翻译成 JVM 字节码,AviatorScript 的基础性能较好。
使用场景包括:
- 规则判断及规则引擎
- 公式计算
- 动态脚本控制
- 集合数据 ELT 等 ……
2.Aviator入门常用
这里使用版本为4.2.5
2.1数据类型
- Number类型: 数字类型,支持四种类型,分别是long,double,java.math.BigInteger(简称 big int)和java.math.BigDecimal(简 称 decimal),规则如下:
- 任何以大写字母 N 结尾的整数都被认为是 big int
- 任何以大写字母 M 结尾的数字都被认为是 decimal
- 其他的任何整数都将被转换为 Long
- 其他任何浮点数都将被转换为 Double
- 超过 long 范围的整数字面量都将自动转换为 big int 类型
- String类型: 字符串类型,单引号或者双引号括起来的文本串,如’hello world’, 变量如果传入的是String或者Character也将转为String类型
- Bool类型: 常量true和false,表示真值和假值,与 java 的Boolean.TRUE和Boolean.False对应
- Pattern类型: 正则表达式, 以//括起来的字符串,如/\d+/,内部 实现为java.util.Pattern
- 变量类型: 与 Java 的变量命名规则相同,变量的值由用户传入
- nil类型: 常量nil,类似 java 中的null,但是nil比较特殊,nil不仅可以参与==、!=的比较, 也可以参与>、>=、<、<=的比较,Aviator 规定任何类型都大于nil除了nil本身,nil==nil返回true。 用户传入的变量值如果为null,那么也将作为nil处理,nil打印为null
2.2操作符
算术运算符
Aviator 支持常见的算术运算符,包括+ - * / %五个二元运算符,和一元运算符-(负)。其中- * / %和一元的-仅能作用于Number类型。
+不仅能用于Number类型,还可以用于String的相加,或者字符串与其他对象的相加。
Aviator 规定,任何类型与String相加,结果为String。
逻辑运算符
Avaitor 的支持的逻辑运算符包括,一元否定运算符!,以及逻辑与的&&,逻辑或的||。逻辑运算符的操作数只能为Boolean。
&&和||都执行短路规则。
关系运算符
Aviator 支持的关系运算符包括<, <=, >, >=以及==和!= 。
关系运算符可以作用于Number之间、String之间、Pattern之间、Boolean之间、变量之间以及其他类型与nil之间的关系比较, 不同类型除了nil之外不能相互比较。
3.一些简单特性
package aviator;import com.googlecode.aviator.AviatorEvaluator;
import com.googlecode.aviator.Expression;
import com.googlecode.aviator.runtime.function.AbstractFunction;
import com.googlecode.aviator.runtime.function.AbstractVariadicFunction;
import com.googlecode.aviator.runtime.function.FunctionUtils;
import com.googlecode.aviator.runtime.type.AviatorDouble;
import com.googlecode.aviator.runtime.type.AviatorObject;
import com.googlecode.aviator.runtime.type.AviatorString;import java.util.HashMap;
import java.util.Map;/*** @author zousy* @version v1.0* @Description* @date 2021-01-16 15:39*/
public class App {public static void main(String[] args) {//执行表达式 Aviator的数值类型仅支持Long和Double, 任何整数都将转换成Long,// 任何浮点数都将转换为Double, 包括用户传入的变量数值Object execute = AviatorEvaluator.execute("1+1+2");System.out.println(execute.toString() + execute.getClass().getSimpleName());Object execute1 = AviatorEvaluator.execute("2.3+3.0");System.out.println(execute1.toString() + execute1.getClass().getSimpleName());//内置函数和 多行表达式 使用; 隔开Expression compile = AviatorEvaluator.getInstance().compile("println('Hello, AviatorScript!'); 1+2");Object res = compile.execute();System.out.println(res.toString());//使用变量Map<String, Object> env = new HashMap<String, Object>();env.put("name","zsad");String execute2 = (String) AviatorEvaluator.execute(" 'hello' + name", env);System.out.println(execute2);//支持函数调用Object execute3 = AviatorEvaluator.execute("string.length('hello')");System.out.println(execute3.toString());//支持通过 lambda 关键字定义一个匿名函数,并且支持闭包捕获//lambda (参数1,参数2...) -> 参数体表达式 endenv.put("x",1);env.put("y",10);Object execute4 = AviatorEvaluator.execute("(lambda (x,y) -> x + y end)(x,y)", env);System.out.println(execute4.toString());//支持自定义函数AviatorEvaluator.addFunction(new AddFunction());AviatorEvaluator.addFunction(new GetFirstNonNullFunction());env.put("a",12);env.put("v",13);//参数确定 自定义函数System.out.println(AviatorEvaluator.execute("add(1,33)").toString());//参数不确定自定义函数System.out.println(AviatorEvaluator.execute("getFirstNonNull(a,v,bas,asf)",env));//lambda表达式 自定义函数AviatorEvaluator.defineFunction("sub","lambda (a,b) -> a - b end");System.out.println(AviatorEvaluator.execute("sub(12,13)").toString());//lambda表达式支持闭包 自定义函数 ? 按照官方写 会报错 存疑AviatorEvaluator.defineFunction("sub2","lambda (a) -> lambda (b) -> a - b end end");System.out.println(AviatorEvaluator.execute("sub2(12)(14)").toString());}//参数确定 继承 AbstractFunctionstatic class AddFunction extends AbstractFunction {@Overridepublic 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());}//函数名 add@Overridepublic String getName() {return "add";}}//参数不确定 继承 AbstractVariadicFunctionstatic class GetFirstNonNullFunction extends AbstractVariadicFunction {@Overridepublic AviatorObject variadicCall(Map<String, Object> env, AviatorObject... args) {if (args != null) {for (AviatorObject arg : args) {if (arg.getValue(env) != null) {return arg;}}}return new AviatorString(null);}@Overridepublic String getName() {return "getFirstNonNull";}}
}
运行结果
4.官方函数例子(官方代码)
package aviator.example;import com.googlecode.aviator.AviatorEvaluator;import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;/*** @author zousy* @version v1.0* @Description* @date 2021-01-18 15:59*/
public class FunctionExample {public static void main(String[] args) {System.out.println(AviatorEvaluator.execute("sysdate()"));System.out.println(AviatorEvaluator.execute("rand()"));System.out.println(AviatorEvaluator.execute("now()"));System.out.println(AviatorEvaluator.execute("date_to_string(sysdate(),'yyyy-MM-dd')"));System.out.println(AviatorEvaluator.execute("string_to_date(date_to_string(sysdate(),'yyyy-MM-dd'),'yyyy-MM-dd')"));// string functionSystem.out.println("test string function...");System.out.println(AviatorEvaluator.execute("string.length('hello')"));System.out.println(AviatorEvaluator.execute("string.contains('hello','h')"));System.out.println(AviatorEvaluator.execute("string.startsWith('hello','h')"));System.out.println(AviatorEvaluator.execute("string.endsWith('hello','llo')"));System.out.println(AviatorEvaluator.execute("string.contains(\"test\",string.substring('hello',1,2))"));System.out.println(Arrays.toString((String[]) AviatorEvaluator.execute("string.split('hello world,aviator',' ')")));// math functionSystem.out.println("test math function...");System.out.println(AviatorEvaluator.execute("math.abs(-3)"));System.out.println(AviatorEvaluator.execute("math.pow(-3,2)"));System.out.println(AviatorEvaluator.execute("math.sqrt(14.0)"));System.out.println(AviatorEvaluator.execute("math.log(100)"));System.out.println(AviatorEvaluator.execute("math.log10(1000)"));System.out.println(AviatorEvaluator.execute("math.sin(20)"));System.out.println(AviatorEvaluator.execute("math.cos(99.23)"));System.out.println(AviatorEvaluator.execute("math.tan(19.9)"));// seq libMap<String, Object> env = new HashMap<String, Object>();ArrayList<Integer> list = new ArrayList<Integer>();list.add(3);list.add(100);list.add(-100);env.put("list", list);System.out.println(AviatorEvaluator.execute("reduce(list,+,0)", env));System.out.println(AviatorEvaluator.execute("filter(list,seq.exists())", env));System.out.println(AviatorEvaluator.execute("count(list)", env));System.out.println(AviatorEvaluator.execute("include(list,100)", env));System.out.println(AviatorEvaluator.execute("sort(list)", env));System.out.println(AviatorEvaluator.execute("map(list,println)", env));System.out.println(list);}
}
运行结果
5.支持大数据操作
package aviator.example;import com.googlecode.aviator.AviatorEvaluator;
import com.googlecode.aviator.Options;import java.math.BigDecimal;
import java.math.BigInteger;
import java.math.MathContext;
import java.util.HashMap;
import java.util.Map;/*** @author zousy* @version v1.0* @Description* @date 2021-01-18 16:10*/
public class BigNumberExample {//以大写字母N为后缀的整数都被认为是big int,如1N,2N,9999999999999999999999N等, 都是big int类型。//超过long范围的整数字面量都将自动转换为big int类型。//以大写字母M为后缀的数字都被认为是decimal, 如1M,2.222M, 100000.9999M等, 都是decimal类型。//big int和decimal的运算,跟其他数字类型long,double没有什么区别,操作符仍然是一样的。// aviator重载了基本算术操作符来支持这两种新类型:public static void main(String[] args) {Object rt = AviatorEvaluator.execute("9223372036854775807100.356M * 2");System.out.println(rt + " " + rt.getClass());rt = AviatorEvaluator.execute("92233720368547758074+1000");System.out.println(rt + " " + rt.getClass());BigInteger a = new BigInteger(String.valueOf(Long.MAX_VALUE) + String.valueOf(Long.MAX_VALUE));BigDecimal b = new BigDecimal("3.2");BigDecimal c = new BigDecimal("9999.99999");Map<String, Object> env = new HashMap<String, Object>();env.put("a", a);env.put("b", b);env.put("c", c);rt = AviatorEvaluator.execute("a+10000000000000000000", env);System.out.println(rt + " " + rt.getClass());rt = AviatorEvaluator.execute("b+c*2", env);System.out.println(rt + " " + rt.getClass());rt = AviatorEvaluator.execute("a*b/c", env);System.out.println(rt + " " + rt.getClass());// set math contextAviatorEvaluator.setOption(Options.MATH_CONTEXT, MathContext.DECIMAL64);rt = AviatorEvaluator.execute("a*b/c", env);System.out.println(rt + " " + rt.getClass());}
}
运行结果
6.正则表达式匹配
package aviator.example;import com.googlecode.aviator.AviatorEvaluator;import java.util.HashMap;
import java.util.Map;/*** @author zousy* @version v1.0* @Description* @date 2021-01-18 16:40*/
public class RegularExpressionExample {//email与正则表达式/([\\w0-8]+@\\w+[\\.\\w+]+)/通过=~操作符来匹配,结果为一个 Boolean 类 型,// 因此可以用于三元表达式判断,匹配成功的时候返回$1,指代正则表达式的分组 1,也就是用户名,否则返回unknown。//Aviator 在表达式级别支持正则表达式,通过//括起来的字符序列构成一个正则表达式,// 正则表达式可以用于匹配(作为=~的右操作数)、比较大小。但是匹配仅能与字符串进行匹配。// 匹配成功后, Aviator 会自动将匹配成功的捕获分组(capturing groups) 放入 env ${num}的变量中,// 其中$0 指代整个匹配的字符串,而$1表示第一个分组,$2表示第二个分组以此类推。//请注意,分组捕获放入 env 是默认开启的,因此如果传入的 env 不是线程安全并且被并发使用,可能存在线程安全的隐患。// 关闭分组匹配,可以通过 AviatorEvaluator.setOption(Options.PUT_CAPTURING_GROUPS_INTO_ENV, false);来关闭,对性能有稍许好处。//Aviator 的正则表达式规则跟 Java 完全一样,因为内部其实就是使用java.util.regex.Pattern做编译的。public static void main(final String[] args) {String email = "killme2008@gmail.com";Map<String, Object> env = new HashMap<>();env.put("email", email);String username =(String) AviatorEvaluator.execute("email=~/([\\w0-8]+)@\\w+[\\.\\w+]+/ ? $1:'unknow'", env);System.out.println(username);}
}
运行结果