137 lines
4.0 KiB
Java
137 lines
4.0 KiB
Java
/**
|
|
* Copyright (c) 2011-2017, James Zhan 詹波 (jfinal@126.com).
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package com.jfinal.template.expr.ast;
|
|
|
|
import java.math.BigDecimal;
|
|
import com.jfinal.template.TemplateException;
|
|
import com.jfinal.template.expr.Sym;
|
|
import com.jfinal.template.stat.Location;
|
|
import com.jfinal.template.stat.ParseException;
|
|
import com.jfinal.template.stat.Scope;
|
|
|
|
/**
|
|
* unary : ('!' | '+' | '-'| '++' | '--') expr
|
|
*
|
|
* 只支持 +expr 与 -expr
|
|
* !expr、 ++expr、 --expr 分别由 Logic、IncDec 支持
|
|
*/
|
|
public class Unary extends Expr {
|
|
|
|
private Sym op;
|
|
private Expr expr;
|
|
|
|
public Unary(Sym op, Expr expr, Location location) {
|
|
if (expr == null) {
|
|
throw new ParseException("The parameter of \"" + op.value() + "\" operator can not be blank", location);
|
|
}
|
|
this.op = op;
|
|
this.expr = expr;
|
|
this.location = location;
|
|
}
|
|
|
|
/**
|
|
* unary : ('!' | '+' | '-'| '++' | '--') expr
|
|
*/
|
|
public Object eval(Scope scope) {
|
|
Object value = expr.eval(scope);
|
|
if (value == null) {
|
|
if (scope.getCtrl().isNullSafe()) {
|
|
return null;
|
|
}
|
|
throw new TemplateException("The parameter of \"" + op.value() + "\" operator can not be blank", location);
|
|
}
|
|
if (! (value instanceof Number) ) {
|
|
throw new TemplateException(op.value() + " operator only support int long float double BigDecimal type", location);
|
|
}
|
|
|
|
switch (op) {
|
|
case ADD:
|
|
return value;
|
|
case SUB:
|
|
Number n = (Number)value;
|
|
if (n instanceof Integer) {
|
|
return Integer.valueOf(-n.intValue());
|
|
}
|
|
if (n instanceof Long) {
|
|
return Long.valueOf(-n.longValue());
|
|
}
|
|
if (n instanceof Float) {
|
|
return Float.valueOf(-n.floatValue());
|
|
}
|
|
if (n instanceof Double) {
|
|
return Double.valueOf(-n.doubleValue());
|
|
}
|
|
if (n instanceof BigDecimal) {
|
|
return ((BigDecimal)n).negate();
|
|
}
|
|
throw new TemplateException("Unsupported data type: " + n.getClass().getName(), location);
|
|
default :
|
|
throw new TemplateException("Unsupported operator: " + op.value(), location);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 如果可能的话,将 Unary 表达式转化成 Const 表达式,类似于 ExprParser.buildMapEntry() 需要这种转化来简化实现
|
|
* 除了可简化程序外,还起到一定的性能优化作用
|
|
*
|
|
* Number : +123 -456 +3.14 -0.12
|
|
* Boolean : !true !false
|
|
*
|
|
* 特别注意:
|
|
* Boolean 的支持并不需要,!true、!false 已在 ExprParser 中被 Logic 表达式接管,在此仅为逻辑上的完备性而添加
|
|
*/
|
|
public Expr toConstIfPossible() {
|
|
if (expr instanceof Const && (op == Sym.SUB || op == Sym.ADD || op == Sym.NOT)) {
|
|
} else {
|
|
return this;
|
|
}
|
|
|
|
Expr ret = this;
|
|
Const c = (Const)expr;
|
|
if (op == Sym.SUB) {
|
|
if (c.isInt()) {
|
|
ret = new Const(Sym.INT, -c.getInt());
|
|
} else if (c.isLong()) {
|
|
ret = new Const(Sym.LONG, -c.getLong());
|
|
} else if (c.isFloat()) {
|
|
ret = new Const(Sym.FLOAT, -c.getFloat());
|
|
} else if (c.isDouble()) {
|
|
ret = new Const(Sym.DOUBLE, -c.getDouble());
|
|
}
|
|
} else if (op == Sym.ADD) {
|
|
if (c.isNumber()) {
|
|
ret = c;
|
|
}
|
|
} else if (op == Sym.NOT) {
|
|
if (c.isBoolean()) {
|
|
ret = c.isTrue() ? Const.FALSE : Const.TRUE;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
public String toString() {
|
|
return op.toString() + expr.toString();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|