This is a modified version of https://github.com/silentmatt/expr-eval 这是基于Matthew Crumley的expr-eval修改而成。
处理数学表达式和有限的字符串、日期
支持的运算符、函数见后表
npm install @nbxx/nb-expr-eval
var Parser = require('@nbxx/nb-expr-eval').Parser;
var parser = new Parser();
var expr = parser.parse('2 * x + 1');
console.log(expr.evaluate({ x: 3 })); // 7
// or
Parser.evaluate('6 * x', { x: 7 }) // 42
核心,包含 parse
方法及一些静态方法用于解析表达式。
构造一个新的 Parser
实例。
构造器可以接受一个可选的 options
参数。
下面的例子会创建一个不允许比较,也不允许逻辑运算符,但允许 in
运算符的解析器:
var parser = new Parser({
operators: {
// 默认开启,也可以手动关闭
add: true,
concatenate: true,
conditional: true,
divide: true,
factorial: true,
multiply: true,
power: true,
remainder: true,
subtract: true,
// 关闭逻辑运算,包含: and, or, not, <, ==, !=, 等等
logical: false,
comparison: false,
// 关闭 in 以及 = 操作符
'in': false,
assignment: false
}
});
把表达式解析为 Expression
对象
new Parser().parse(expression)
的静态等价方法
使用 variables
对象中的变量和方法来解析、计算表达式
Parser.evaluate(expr, vars)
等价于 Parser.parse(expr).evaluate(vars).
Parser.parse(str)
返回一个 Expression
对象, Expression
对象与 js 函数很相似,实际上也可以被直接转换为函数。
使用 variables
中的变量来计算 expression
对象, 如果有变量未被解析,则抛出异常。
js> expr = Parser.parse("2 ^ x");
(2^x)
js> expr.evaluate({ x: 3 });
8
使用 expression
来替换原有 expression
中的 variable
。
js> expr = Parser.parse("2 * x + 1");
((2*x)+1)
js> expr.substitute("x", "4 * x");
((2*(4*x))+1)
js> expr2.evaluate({ x: 3 });
25
使用 variables
来替换表达式中的变量,从而简化表达式。函数不会被替换和简化。
实际上simplify只是简单的把变量替换了一下,然后将常数直接计算出结果。
所以 ((2*(4*x))+1)
是没法直接简化的,除非替换 x 。但是 2*4*x+1
是可以简化的,
因为它等价于(((2*4)*x)+1)
, 所以 (2*4)
会被简化为 "8", 生成结果 ((8*x)+1)
.
js> expr = Parser.parse("x * (y * atan(1))").simplify({ y: 4 });
(x*3.141592653589793)
js> expr.evaluate({ x: 2 });
6.283185307179586
列出当前表达式中尚未被简化替换的变量。.
js> expr = Parser.parse("x * (y * atan(1))");
(x*(y*atan(1)))
js> expr.variables();
x,y
js> expr.simplify({ y: 4 }).variables();
x
variables
默认只返回最顶层的变量,例如 Parser.parse(x.y.z).variables()
返回 ['x']
。
如果希望获得细节,可以使用 { withMembers: true }
,随后 Parser.parse(x.y.z).variables({ withMembers: true })
将会返回['x.y.z']
.
获取尚未简化的变量,以及所有函数
js> expr = Parser.parse("min(x, y, z)");
(min(x, y, z))
js> expr.symbols();
min,x,y,z
js> expr.simplify({ y: 4, z: 5 }).symbols();
min,x
与 variables
相似, symbols
接受可选参数 { withMembers: true }
来显示对象成员.
将表达式转化为字符串 toString()
将自动添加括号。.
把 Expression
转换为js函数。 parameters
是参数名列表(Array),或者参数名字符串(逗号分隔的列表)
如果提供了可选参数 variables
表达式将会被简化。
js> expr = Parser.parse("x + y + z");
((x + y) + z)
js> f = expr.toJSFunction("x,y,z");
[Function] // function (x, y, z) { return x + y + z; };
js> f(1, 2, 3)
6
js> f = expr.toJSFunction("y,z", { x: 100 });
[Function] // function (y, z) { return 100 + y + z; };
js> f(2, 3)
105
基本上和js语法差不多,除了部分运算符为数学表达式外。例如: ^
运算符代表指数函数,而不是异或。
运算符 | 相依性 | 说明 |
---|---|---|
(...) | None | 分组 |
f(), x.y, a[i] | Left | 方法调用、成员变量访问、数组下标访问 |
! | Left | 阶乘 |
^ | Right | 乘方 |
+, -, not, sqrt, etc. | Right | 一元前置运算符 (列表见后文) |
*, /, % | Left | 乘以、除以、取余 |
+, -, || | Left | 加、减、列表合并 |
==, !=, >=, <=, >, <, in | Left | 等于、不等于、大于等于、小于等于、大于、小于、"in" 作为中间运算符,代表左侧操作数在右侧操作数之内 |
and | Left | 逻辑 AND |
or | Left | 逻辑 OR |
x ? y : z | Right | 三元运算符 (if x then y else z) |
= | Right | 变量赋值 |
; | Left | 表达式分隔符 |
"in" 和 = 两个运算符默认关闭,需要显式开启才能使用
const parser = new Parser({
operators: {
'in': true,
'assignment': true
}
});
// 现在可以使用 'x in array' 和 'y = 2*x' 两种表达式了
解析器包含一些内置函数,实际上会作为一元运算符来使用。他们和预定义函数的区别是他们只接受一个参数,
而且不需要括号包围起来。包含括号的一元运算符优先级和函数一致,不包含括号的一元运算符优先级按运算符优先级来计算(低于 '^')。
例如,sin(x)^2
等价于 (sin x)^2
, 而 sin x^2
等价于 sin(x^2)
。
+
和 -
两种一元运算符是例外,因为不存在对应的函数,所以优先级始终按运算符优先级来计算。
运算符 | 说明 |
---|---|
-x | 负数 |
+x | 一元加号。可以将一个操作数转为数字类型 |
x! | 对于正整数是阶乘,对于非正整数是 gamma(x + 1) 。 |
abs x | x 的绝对值 |
acos x | 反余弦 x (in radians) |
acosh x | 反双曲余弦 x (in radians) |
asin x | 反正弦 x (in radians) |
asinh x | 反双曲正弦 x (in radians) |
atan x | 反正切 x (in radians) |
atanh x | 反双曲正切 x (in radians) |
cbrt x | 开三次方 x |
ceil x | 向上取整 x, 大于等于 x 的最小整数 |
cos x | 余弦 x (x is in radians) |
cosh x | 双曲余弦 x (x is in radians) |
exp x | 以 e 为底数的指数/反对数函数,等价于 e^x |
expm1 x | e^x - 1 |
floor x | 向下取整 x,小于等于 x 的最大的整数 |
length x | x 的字符串长度 |
ln x | x 的自然对数 |
log x | x 的自然对数 |
log10 x | x 的对数 (底数为 10 的对数) |
log2 x | x 的对数 (底数为 2 的对数) |
log1p x | (1+x) 的自然对数 |
not x | 逻辑 NOT |
round x | 舍入取整,使用四舍五入法 |
sign x | x 的正负。-1 代表负数、0 代表 0、1 代表 正数 |
sin x | 正弦 x (x is in radians) |
sinh x | 双曲正弦 x (x is in radians) |
sqrt x | 平方根,如果 x 为负数, 结果为 NAN |
tan x | 正切 x (x is in radians) |
tanh x | 双曲正切 x (x is in radians) |
trunc x | 直接取整,舍去小数部分。x 为正数时向下取整, x 为负数时向上取整 |
除了运算符外,还有一些预定义的函数。预定义函数不会被 simplify 处理
函数 | 说明 |
---|---|
random(n) | 获取 [0,n) 之间的随机数,如果 n = 0,或者未提供 n ,则默认使用 1 代替 n |
fac(n) | n! (factorial of n: "n * (n-1) * (n-2) * … * 2 * 1") Deprecated. Use the ! operator instead. |
min(a,b,…) | 列表中最小的数字 |
max(a,b,…) | 列表中最大的数字 |
hypot(a,b) | 勾股定理。斜边长, a, b 分别指直角三角形的两个直角边,结果是直角三角形斜边长. |
pyt(a, b) | 等价于 hypot. |
pow(x, y) | 等价于 x^y |
atan2(y, x) | atan( x/y ). i.e. 坐标系中点 (0, 0) 到点 (x, y) 连线与x轴之间夹角. ( result is in radians) |
roundTo(x, n) | 将 x 四舍五入 n 位小数. |
map(f, a) | 列表映射。遍历列表 a 的每个元素 x,并计算 f(x) 作为新列表的元素值 |
fold(f, y, a) | 列表汇聚。遍历列表 a 的每个元素 x、序号 i,给定一个初始值,计算 f(y, x, i) 作为新列表的元素。最后计算新列表的和。 |
filter(f, a) | 列表筛选。遍历列表 a 的每个元素 x、序号 i,只保留 f(x, i) 为 true 的元素 |
indexOf(x, a) | 返回列表/字符串中第一个匹配的元素的下标,没有找到则返回 -1 |
join(sep, a) | 使用 sep 拼接 列表 a 中的全部字符串元素 |
if(c, a, b) | 三元运算符 c ? a : b 的 function 形式。但总会计算全部分支,如果对性能有要求应该直接使用三元运算符。 |
substr(str, start, length) | 获取字符串 str 中从 start 开始持续 length 个字符,如果不填 length 则获取从 start 开始到字符串结尾 |
列表能直接使用 [] 定义,例如: [ 1, 2, 3, 2+2, 10/2, 3! ]
可以使用 name(params) = expression
的格式来自定义函数。函数支持多参数
Examples:
square(x) = x*x
add(a, b) = a + b
factorial(x) = x < 2 ? 1 : x * factorial(x - 1)
parser 有一个属性 functions ,记录了所有的函数。如果想要在全局的角度自定义函数,只需要添加即可
const parser = new Parser();
// Add a new function
parser.functions.customAddFunction = function (arg1, arg2) {
return arg1 + arg2;
};
// Remove the factorial function
delete parser.functions.fac;
parser.evaluate('customAddFunction(2, 4) == 6'); // true
//parser.evaluate('fac(3)'); // This will fail
以下是预定义的常量:
常量 | 说明 |
---|---|
E | 等价于 js 的 Math.E |
PI | 等价于 js 的 Math.PI |
true | 逻辑 true |
false | 逻辑 false |
与函数类似,预定义常量也保存在 parser 属性中。可以随意在 parser.consts
中添加自定义的常量。比如:
const parser = new Parser();
parser.consts.R = 1.234;
console.log(parser.parse('A+B/R').toString()); // ((A + B) / 1.234)
To disable the pre-defined constants, you can replace or delete parser.consts
:
const parser = new Parser();
parser.consts = {};
cd
to the project directory- Install development dependencies:
npm install
- Run the tests:
npm test