-
Notifications
You must be signed in to change notification settings - Fork 863
Home (English) (Draft 草稿 Incomplete 不完整)
Created by gh-md-toc
Aviator is a lightweight, high performance, expression evaluation engine for Java, used mainly for the evaluation of various dynamic expressions. There are numerous open-source expression evaluation engines for Java already available, and thus, the question: Why use Aviator?
Aviator's design goals are to be lightweight and provide high performance: Unlike Groovy and JRuby, Aviator is not bulky, totalling only 450K with its dependencies, and only 70K without. Of course, Aviator's syntax is limited; It covers just a small section of the language, and isn't comprehensive.
其次, Aviator的实现思路与其他轻量级的求值器很不相同, 其他求值器一般都是通过解释的方式运行, 而Aviator则是直接将表达式编译成Java 字节码, 交给JVM去执行。简单来说, Aviator的定位是介于Groovy这样的重量级脚本语言和IKExpression这样的轻量级表达式引擎 之间。
Aviator支持大部分运算操作符, 包括算术操作符、关系运算符、逻辑操作符、位运算符、正则匹配操作符(=~
)、三元表达式(?:
), 并且支持操作符的优先级和括号强制优先级, 具体请看后面的操作符列表, 支持自定义函数.
Aviator depends on commons-beanutils
, 使用Aviator可以添加下面的maven依赖:
<dependency>
<groupId>com.googlecode.aviator</groupId>
<artifactId>aviator</artifactId>
<version>2.3.4</version>
</dependency>
Aviator的使用都是集中通过com.googlecode.aviator.AviatorEvaluator
这个入口类来处理, 最简单的例子, 执行一个计算1+2+3
的表达式:
import com.googlecode.aviator.AviatorEvaluator;
public class TestAviator {
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
, 包括用户传入的变量数值。这个例子的打印结果将是正确答案6
。
例外情况是,如果开启了 ALWAYS_USE_DOUBLE_AS_DECIMAL
选项,那么在表达式中出现的浮点数都将解析为 BigDecimal,这是为了方便一些用户要求高精度的计算,又不想额外地给浮点数加上 M
后缀标记为 BigDecimal:
AviatorEvaluator.setOption(Options.ALWAYS_USE_DOUBLE_AS_DECIMAL, true);
Want Aviator to say hello to you? This is very easy to do. Pass in your name so that Aviator can add it to the string.
public class TestAviator {
public static void main(String[] args) {
String yourName = "Michael";
Map<String, Object> env = new HashMap<String, Object>();
env.put("yourName", yourName);
String result = (String) AviatorEvaluator.execute(" 'hello ' + yourName ", env);
System.out.println(result); // hello Michael
}
}
上面的例子演示了怎么向表达式传入变量值, 表达式中的yourName
是一个变量, 默认为null
, 通过传入Map<String,Object>
的变量绑定环境, 将yourName
设置为你输入的名称。 env
的key
是变量名, value
是变量的值。
上面例子中的'hello '
是一个Aviator
的String
, Aviator
的String
是任何用单引号或者双引号括起来的字符序列, String
可以比较大小(基于unicode
顺序), 可以参与正则匹配, 可以与任何对象相加, 任何对象与String
相加结果为String
。 String
中也可以有转义字符,如\n、\\、\'
等。
AviatorEvaluator.execute(" 'a\"b' "); // 字符串 a"b
AviatorEvaluator.execute(" \"a\'b\" "); // 字符串 a'b
AviatorEvaluator.execute(" 'hello ' + 3 "); // 字符串 hello 3
AviatorEvaluator.execute(" 'hello '+ unknow "); // 字符串 hello null
An exec
method is included as of Aviator 2.2, providing a more convenient way to pass in and execute variables, without the need to construct env
in the map
:
String name = "dennis";
AviatorEvaluator.exec(" 'hello ' + yourName ", name); // hello dennis
只要在exec
中按照变量在表达式中的出现顺序传入变量值就可以执行, 不需要构建Map
了。
Aviator supports function calls in a similar style to that of lua. The following example gets the length of a string:
AviatorEvaluator.execute("string.length('hello')"); // 5
string.length('hello')
是一个函数调用, string.length
是一个函数, 'hello'
是调用的参数。
再用string.substring
来截取字符串:
AviatorEvaluator.execute("string.contains(\"test\", string.substring('hello', 1, 2))"); // true
The output of string.substring('hello', 1, 2)
is passed to 'e'
, and then it's determined whether 'test'
contains the value of e
by using string.contains
. As you can see, functions can be nested. Refer to aviator's built-in functions list for more information.
In addition to Aviator's built-in functions, custom functions are also supported and can be used, as long as com.googlecode.aviator.runtime.type.AviatorFunction
is implemented and registered to the AviatorEvaluator
interface. The AviatorFunction
interface is very large. Usually you don't need to implement all methods, Just base the number of your method's parameters 只要根据你的方法的参 数个数, 继承AbstractFunction
类并override
相应方法即可。
可以看一个例子,我们实现一个add
函数来做数值的相加:
public class TestAviator {
public static void main(String[] args) {
//Register function.
AviatorEvaluator.addFunction(new AddFunction());
System.out.println(AviatorEvaluator.execute("add(1, 2)")); // 3.0
System.out.println(AviatorEvaluator.execute("add(add(1, 2), 100)")); // 103.0
}
}
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";
}
}
注册函数通过AviatorEvaluator.addFunction
方法, 移除可以通过removeFunction
。另外, FunctionUtils 提供了一些方便参数类型转换的方法。
如果你的参数个数不确定,可以继承 AbstractVariadicFunction
类,只要实现其中的 variadicCall
方法即可,比如我们实现一个找到第一个参数不为 null 的函数:
public class GetFirstNonNullFunction extends AbstractVariadicFunction {
public 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);
}
@Override
public String getName() {
return "getFirstNonNull";
}
}
注册后使用就可以传入不定参数了:
getFirstNonNull(1);
getFirstNonNull(1,2,3,4,null,5);
getFirstNonNull(a,b,c,d);
当然,同时你仍然覆写特定的 call
方法来自定义实现。
In addition to adding custom functions by way of AviatorEvaluator.addFunction
, you can add a configuration file under classpath by way of aviator_functions.config
. Content is a list of lines containing the complete names of the custom functions. For example:
# This is a comment line.
com.example.TestFunction
com.example.GetFirstNonNullFunction
那么 Aviator 将在 JVM 启动的时候自动加载这些自定义函数,配置文件中以 #
开头的行将被认为是注释。如果你想自定义文件路径,可以通过传入环境变量
-Dcom.googlecode.aviator.custom_function_config_file=xxxx.config
To install.
上面提到的例子都是直接执行表达式, 事实上 Aviator 背后都帮你做了编译并执行的工作。 你可以自己先编译表达式, 返回一个编译的结果, 然后传入不同的env
来复用编译结果, 提高性能, 这是更推荐的使用方式:
public class TestAviator {
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.3);
env.put("b", 45);
env.put("c", -199.100);
// Execute expressions
Boolean result = (Boolean) compiledExp.execute(env);
System.out.println(result); // false
}
}
通过compile
方法可以将表达式编译成Expression
的中间对象, 当要执行表达式的时候传入env
并调用Expression
的execute
方法即可。 表达式中使用了括号来强制优先级, 这个例子还使用了>
用于比较数值大小, 比较运算符!=、==、>、>=、<、<=
不仅可以用于数值, 也可以用于String、Pattern、Boolean
等等, 甚至是任何用户传入的两个都实现了java.lang.Comparable
接口的对象之间。
编译后的结果你可以自己缓存, 也可以交给 Aviator 帮你缓存, AviatorEvaluator
内部有一个全局的缓存池, 如果你决定缓存编译结果, 可以通过:
public static Expression compile(String expression, boolean cached)
将cached
设置为true
即可, 那么下次编译同一个表达式的时候将直接返回上一次编译的结果。
使缓存失效通过:
public static void invalidateCache(String expression)
方法。
可以通过中括号去访问数组和java.util.List
对象, 可以通过map.key
访问java.util.Map
中key
对应的value
, 一个例子:
public static void main(String[] args) {
final List<String> list = new ArrayList<String>();
list.add("hello");
list.add(" world");
final int[] array = new int[3];
array[0] = 0;
array[1] = 1;
array[2] = 3;
final Map<String, Date> map = new HashMap<String, Date>();
map.put("date", new Date());
Map<String, Object> env = new HashMap<String, Object>();
env.put("list", list);
env.put("array", array);
env.put("mmap", map);
System.out.println(AviatorEvaluator.execute("list[0]+list[1]", env)); // hello world
System.out.println(AviatorEvaluator.execute("'array[0]+array[1]+array[2]=' + (array[0]+array[1]+array[2])", env)); // array[0]+array[1]+array[2]=4
System.out.println(AviatorEvaluator.execute("'today is ' + mmap.date ", env)); // today is Wed Feb 24 17:31:45 CST 2016
}
如果函数调用或者括号表达式结果是一个数组或者List,你同样可以可以通过 [index]
访问:
assertEquals("a", AviatorEvaluator.exec("string.split(s,',')[0]", "a,b,c,d"));
Aviator 不提供if else
语句, 但是提供了三元操作符?:
用于条件判断,使用上与 java 没有什么不同:
AviatorEvaluator.exec("a>0? 'yes':'no'", 1); // yes
Aviator 的三元表达式对于两个分支的结果类型并不要求一致,可以是任何类型,这一点与 java 不同。
Aviator 支持类 Ruby 和 Perl 风格的表达式匹配运算,通过=~
操作符, 如下面这个例子匹配 email 并提取用户名返回:
public static void main(String[] args) {
String email = "[email protected]";
Map<String, Object> env = new HashMap<String, Object>();
env.put("email", email);
String username = (String) AviatorEvaluator.execute("email=~/([\\w0-8]+)@\\w+[\\.\\w+]+/ ? $1 : 'unknow' ", env);
System.out.println(username); // killme2008
}
email
与正则表达式/([\\w0-8]+@\\w+[\\.\\w+]+)/
通过=~
操作符来匹配,结果为一个 Boolean
类 型, 因此可以用于三元表达式判断,匹配成功的时候返回$1
,指代正则表达式的分组 1,也就是用户名,否则返回unknown
。
Aviator 在表达式级别支持正则表达式,通过//
括起来的字符序列构成一个正则表达式,正则表 达式可以用于匹配(作为=~
的右操作数)、比较大小,匹配仅能与字符串进行匹配。匹配成功后, Aviator 会自动将匹配成功的分组放入$num
的变量中,其中$0
指代整个匹配的字符串,而$1
表示第一个分组,以此类推。
Aviator 的正则表达式规则跟 Java 完全一样,因为内部其实就是使用java.util.regex.Pattern
做编译的。
Aviator 有个方便用户Using variables的语法糖, 当你要访问变量a
中的某个属性b
, 那么你可以通过a.b
访问到, 更进一步, a.b.c
将访问变量a
的b
属性中的c
属性值, 推广开来也就是说 Aviator 可以将变量声明为嵌套访问的形式。
TestAviator
类符合JavaBean
规范, 并且是 public
的,我们就可以使用语法糖:
public class TestAviator {
int i;
float f;
Date date;
// 构造方法
public TestAviator(int i, float f, Date date) {
this.i = i;
this.f = f;
this.date = date;
}
// getter and setter
public static void main(String[] args) {
TestAviator foo = new TestAviator(100, 3.14f, new Date());
Map<String, Object> env = new HashMap<String, Object>();
env.put("foo", foo);
System.out.println(AviatorEvaluator.execute("'foo.i = '+foo.i", env)); // foo.i = 100
System.out.println(AviatorEvaluator.execute("'foo.f = '+foo.f", env)); // foo.f = 3.14
System.out.println(AviatorEvaluator.execute("'foo.date.year = '+(foo.date.year+1990)", env)); // foo.date.year = 2106
}
}
nil
是 Aviator 内置的常量,类似 java 中的null
,表示空的值。nil
跟null
不同的在于,在 java 中null
只能使用在==、!=
的比较运算符,而nil
还可以使用>、>=、<、<=
等比较运算符。 Aviator 规定,任何对象都比nil
大除了nil
本身。用户传入的变量如果为null
,将自动以nil
替代。
AviatorEvaluator.execute("nil == nil"); //true
AviatorEvaluator.execute(" 3> nil"); //true
AviatorEvaluator.execute(" true!= nil"); //true
AviatorEvaluator.execute(" ' '>nil "); //true
AviatorEvaluator.execute(" a==nil "); //true, a 是 null
nil
与String
相加的时候,跟 java 一样显示为 null
Aviator 并不支持日期类型,如果要比较日期,你需要将日期写字符串的形式,并且要求是形如 “yyyy-MM-dd HH:mm:ss:SS”的字符串,否则都将报错。 字符串跟java.util.Date
比较的时候将自动转换为Date
对象进行比较:
public static void main(String[] args) {
Map<String, Object> env = new HashMap<String, Object>();
final Date date = new Date();
String dateStr = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SS").format(date);
env.put("date", date);
env.put("dateStr", dateStr);
Boolean result = (Boolean) AviatorEvaluator.execute("date==dateStr", env);
System.out.println(result); // true
result = (Boolean) AviatorEvaluator.execute("date > '2010-12-20 00:00:00:00' ", env);
System.out.println(result); // true
result = (Boolean) AviatorEvaluator.execute("date < '2200-12-20 00:00:00:00' ", env);
System.out.println(result); // true
result = (Boolean) AviatorEvaluator.execute("date==date ", env);
System.out.println(result); // true
}
也就是说String
除了能跟String
比较之外,还能跟nil
和java.util.Date
对象比较。
从 2.3.0 版本开始,aviator 开始支持大数字计算和特定精度的计算, 本质上就是支持java.math.BigInteger
和java.math.BigDecimal
两种类型, 这两种类型在 aviator 中简称 为big int
和decimal
类型。 类似99999999999999999999999999999999
这样的数字在 Java 语言里是没办法编译通过 的, 因为它超过了Long
类型的范围, 只能用BigInteger
来封装。但是 aviator 通过包装,可 以直接支持这种大整数的计算,例如:
public static void main(String[] args) {
System.out.println(AviatorEvaluator.exec("99999999999999999999999999999999 + 99999999999999999999999999999999"));
}
结果为类型big int
的: 199999999999999999999999999999998
big int
和decimal
的表示与其他数字不同,两条规则:
- 以大写字母
N
为后缀的整数都被认为是big int
,如1N,2N,9999999999999999999999N
等, 都是big int
类型。 - 超过
long
范围的整数字面量都将自动转换为big int
类型。 - 以大写字母
M
为后缀的数字都被认为是decimal
, 如1M,2.222M, 100000.9999M
等, 都是decimal
类型。
用户也可以通过变量传入这两种类型来参与计算。
如果用户觉的给浮点数添加 M
后缀比较繁琐,也可以强制所有浮点数解析为 BigDecimal,通过代码开启下列选项即可:
AviatorEvaluator.setOption(Options.ALWAYS_USE_DOUBLE_AS_DECIMAL, true);
big int
和decimal
的运算,跟其他数字类型long,double
没有什么区别,操作符仍然是一样的。 aviator重载了基本算术操作符来支持这两种新类型:
public static void main(String[] args) {
Object rt = AviatorEvaluator.exec("9223372036854775807100.356M * 2");
System.out.println(rt + " " + rt.getClass()); // 18446744073709551614200.712 class java.math.BigDecimal
rt = AviatorEvaluator.exec("92233720368547758074+1000");
System.out.println(rt + " " + rt.getClass()); // 92233720368547759074 class java.math.BigInteger
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");
rt = AviatorEvaluator.exec("a+10000000000000000000", a);
System.out.println(rt + " " + rt.getClass()); // 92233720368547758089223372036854775807 class java.math.BigInteger
rt = AviatorEvaluator.exec("b+c*2", b, c);
System.out.println(rt + " " + rt.getClass()); // 20003.19998 class java.math.BigDecimal
rt = AviatorEvaluator.exec("a*b/c", a, b, c);
System.out.println(rt + " " + rt.getClass()); // 2.951479054745007313280155218459508E+34 class java.math.BigDecimal
}
当big int
或者decimal
和其他类型的数字做运算的时候,按照long < big int < decimal < double
的规则做提升, 也就是说运算的数字如果类型不一致, 结果的类型为两者之间更“高”的类型。例如:
-
1 + 3N
, 结果为big int
的4N
-
1 + 3.1M
,结果为decimal
的4.1M
-
1N + 3.1M
,结果为decimal
的4.1M
-
1.0 + 3N
,结果为double
的4.0
-
1.0 + 3.1M
,结果为double
的4.1
Java 的java.math.BigDecimal
通过java.math.MathContext
支持特定精度的计算,任何涉及到金额的计算都应该使用decimal
类型。
默认 Aviator 的计算精度为MathContext.DECIMAL128
,你可以自定义精度, 通过:
AviatorEvaluator.setOption(Options.MATH_CONTEXT, MathContext.DECIMAL64);
即可设置,更多关于decimal
的精度问题请看java.math.BigDecimal
的 javadoc 文档。
aviator 拥有强大的操作集合和数组的 seq
库。整个库风格类似函数式编程中的高阶函数。在 aviator 中, 数组以及java.util.Collection
下的子类都称为seq
,可以直接利用 seq
库进行遍历、过滤和聚合等操作。
例如,假设我有个 list
:
public static void main(String[] args) {
Map<String, Object> env = new HashMap<String, Object>();
ArrayList<Integer> list = new ArrayList<Integer>();
list.add(3);
list.add(20);
list.add(10);
env.put("list", list);
Object result = AviatorEvaluator.execute("count(list)", env);
System.out.println(result); // 3
result = AviatorEvaluator.execute("reduce(list,+,0)", env);
System.out.println(result); // 33
result = AviatorEvaluator.execute("filter(list,seq.gt(9))", env);
System.out.println(result); // [10, 20]
result = AviatorEvaluator.execute("include(list,10)", env);
System.out.println(result); // true
result = AviatorEvaluator.execute("sort(list)", env);
System.out.println(result); // [3, 10, 20]
AviatorEvaluator.execute("map(list,println)", env);
}
我们可以:
- 求长度:
count(list)
- 求和:
reduce(list,+,0)
,reduce
函数接收三个参数,第一个是seq
,第二个是聚合的函数,如+
等,第三个是聚合的初始值 - 过滤:
filter(list,seq.gt(9))
, 过滤出list
中所有大于9
的元素并返回集合;seq.gt
函数用于生成一个谓词,表示大于某个值 - 判断元素在不在集合里:
include(list,10)
- 排序:
sort(list)
- 遍历整个集合:
map(list,println)
,map
接受的第二个函数将作用于集合中的每个元素,这里简单地调用println
打印每个元素
默认 AviatorEvaluator
以执行速度优先:
AviatorEvaluator.setOption(Options.OPTIMIZE_LEVEL, AviatorEvaluator.EVAL);
你可以修改为编译速度优先,这样不会做编译优化:
AviatorEvaluator.setOption(Options.OPTIMIZE_LEVEL, AviatorEvaluator.COMPILE);
如果想查看每个表达式生成的字节码,可以通过打开 Trace 选项:
import com.googlecode.aviator.Options;
......
AviatorEvaluator.setOption(Options.TRACE, true);
默认是打印到标准输出,你可以改变输出指向:
AviatorEvaluator.setTraceOutputStream(new FileOutputStream(new File("aviator.log")));
下面是 Aviator 详细的语法规则定义。
-
Number
类型: 数字类型,支持四种类型,分别是long,double,java.math.BigInteger(简称 big int)
和java.math.BigDecimal(简 称 decimal)
,规则如下:- 任何以大写字母
N
结尾的整数都被认为是big int
- 任何以大写字母
M
结尾的数字都被认为是decimal
- 其他的任何整数都将被转换为
Long
- 其他任何浮点数都将被转换为
Double
- 超过
long
范围的整数字面量都将自动转换为big int
类型
- 任何以大写字母
其中 big int
和 decimal
是 2.3.0 版本开始引入的。数字还支持十六进制(以0x
或者0X
开头的数字), 以及科学计数法,如1e-3
等。 不支持其他进制。
-
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
Aviator 支持常见的算术运算符,包括+ - * / %
五个二元运算符,和一元运算符-(负)
。其中- * / %
和一元的-
仅能作用于Number
类型。
+
不仅能用于Number
类型,还可以用于String
的相加,或者字符串与其他对象的相加。
Aviator 规定,任何类型与String
相加,结果为String
。
Avaitor 的支持的逻辑运算符包括,一元否定运算符!
,以及逻辑与的&&
,逻辑或的||
。逻辑运算符的操作数只能为Boolean
。
&&
和||
都执行短路规则。
Aviator 支持的关系运算符包括<, <=, >, >=
以及==
和!=
。
关系运算符可以作用于Number
之间、String
之间、Pattern
之间、Boolean
之间、变量之间以及其他类型与nil
之间的关系比较, 不同类型除了nil
之外不能相互比较。
Aviator 支持所有的 Java 位运算符,包括&, |, ^, ~, >>, <<, >>>
。
匹配运算符=~
用于String
和Pattern
的匹配,它的左操作数必须为String
,右操作数必须为Pattern
。 匹配成功后,Pattern
的分组将存于变量$num
,num
为分组索引。
Aviator 没有提供if else
语句,但是提供了三元运算符?:
,形式为bool ? exp1: exp2
。 其中bool
必须为Boolean
类型的表达式, 而exp1
和exp2
可以为任何合法的 Aviator 表达式,并且不要求exp1
和exp2
返回的结果类型一致。
Function name | Description |
---|---|
sysdate() | Returns the current date object ( java.util.Date ). |
rand() | Returns a random number between 0-1 (double type). |
rand(n) | Returns a random number between 0-n (long type). |
print([out],obj) | 打印对象,如果指定 out,向 out 打印, 否则输出到控制台 |
println([out],obj) | 与 print 类似,但是在输出后换行 |
now() | 返回 System.currentTimeMillis |
long(v) | 将值的类型转为 long |
double(v) | 将值的类型转为 double |
str(v) | 将值的类型转为 string |
date_to_string(date,format) | 将 Date 对象转化化特定格式的字符串,2.1.1 新增 |
string_to_date(source,format) | 将特定格式的字符串转化为 Date 对 象,2.1.1 新增 |
string.contains(s1,s2) | 判断 s1 是否包含 s2,返回 Boolean |
string.length(s) | 求字符串长度,返回 Long |
string.startsWith(s1,s2) | s1 是否以 s2 开始,返回 Boolean |
string.endsWith(s1,s2) | s1 是否以 s2 结尾,返回 Boolean |
string.substring(s,begin[,end]) | 截取字符串 s,从 begin 到 end,如果忽略 end 的话,将从 begin 到结尾,与 java.util.String.substring 一样。 |
string.indexOf(s1,s2) | java 中的 s1.indexOf(s2),求 s2 在 s1 中 的起始索引位置,如果不存在为-1 |
string.split(target,regex,[limit]) | Java 里的 String.split 方法一致,2.1.1 新增函数 |
string.join(seq,seperator) | 将集合 seq 里的元素以 seperator 为间隔 连接起来形成字符串,2.1.1 新增函数 |
string.replace_first(s,regex,replacement) | Java 里的 String.replaceFirst 方法, 2.1.1 新增 |
string.replace_all(s,regex,replacement) | Java 里的 String.replaceAll 方法 , 2.1.1 新增 |
math.abs(d) | 求 d 的绝对值 |
math.sqrt(d) | 求 d 的平方根 |
math.pow(d1,d2) | 求 d1 的 d2 次方 |
math.log(d) | 求 d 的自然对数 |
math.log10(d) | 求 d 以 10 为底的对数 |
math.sin(d) | 正弦函数 |
math.cos(d) | 余弦函数 |
math.tan(d) | 正切函数 |
map(seq,fun) | 将函数 fun 作用到集合 seq 每个元素上, 返回新元素组成的集合 |
filter(seq,predicate) | 将谓词 predicate 作用在集合的每个元素 上,返回谓词为 true 的元素组成的集合 |
count(seq) | 返回集合大小 |
include(seq,element) | 判断 element 是否在集合 seq 中,返回 boolean 值 |
sort(seq) | 排序集合,仅对数组和 List 有效,返回排 序后的新集合 |
reduce(seq,fun,init) | fun 接收两个参数,第一个是集合元素, 第二个是累积的函数,本函数用于将 fun 作用在集合每个元素和初始值上面,返回 最终的 init 值 |
seq.eq(value) | Returns a predicate that evaluates whether equal to value. Used for filter functions, such as filter(seq,seq.eq(3)), which returns a collection of elements equal to 3. |
seq.neq(value) | Returns a predicate that evaluates whether NOT equal to value. |
seq.gt(value) | Returns a predicate that evaluates whether greater than value. |
seq.ge(value) | Returns a predicate that evaluates whether greater than or equal to value. |
seq.lt(value) | Returns a predicate that evaluates whether less than value. |
seq.le(value) | Returns a predicate that evaluates whether less than or equal to value. |
seq.nil() | Returns a predicate that evaluates whether it's nil. |
seq.exists() | Returns a predicate that evaluates whether it's NOT nil. |