问题现象
我们知道MyBatis通过OGNL表达式的方式来进行动态SQL的拼接,比如像下图所示:if
标签代表判断逻辑,但你是否了解,过果type类型为数值时,那么对于type != ''
部分判断会出现一些问题,比如,当type为0
时,type != ''
结果为false
,而当type为非0
数值时type != ''
结果为true
xml
<if test="type != null and type != ''">
</if>
问题分析
此类问题只能通过翻阅源码了解原因。
首先找到MyBatis中对于if
标签的解析部分,如下图所示:
java
public class IfSqlNode implements SqlNode {
private final ExpressionEvaluator evaluator;
private final String test;
private final SqlNode contents;
public IfSqlNode(SqlNode contents, String test) {
this.test = test;
this.contents = contents;
this.evaluator = new ExpressionEvaluator();
}
@Override
public boolean apply(DynamicContext context) {
if (evaluator.evaluateBoolean(test, context.getBindings())) {
contents.apply(context);
return true;
}
return false;
}
}
判断逻辑在evaluateBoolean
方法中,其中参数expression
就是"type != null and type != ''"
java
public boolean evaluateBoolean(String expression, Object parameterObject) {
Object value = OgnlCache.getValue(expression, parameterObject);
if (value instanceof Boolean) {
return (Boolean) value;
}
if (value instanceof Number) {
return new BigDecimal(String.valueOf(value)).compareTo(BigDecimal.ZERO) != 0;
}
return value != null;
}
继续查看OgnlCache.getValue(expression, parameterObject)
方法
java
public static Object getValue(String expression, Object root) {
try {
Map context = Ognl.createDefaultContext(root, MEMBER_ACCESS, CLASS_RESOLVER, null);
return Ognl.getValue(parseExpression(expression), context, root);
} catch (OgnlException e) {
throw new BuilderException("Error evaluating expression '" + expression + "'. Cause: " + e, e);
}
}
private static Object parseExpression(String expression) throws OgnlException {
Object node = expressionCache.get(expression);
if (node == null) {
node = Ognl.parseExpression(expression);
expressionCache.put(expression, node);
}
return node;
}
关键解析在Ognl.getValue(parseExpression(expression), context, root)
方法,先看parseExpression(expression)
方法,这个方法实际上就是解析了"type != null and type != ''"
这段内容,然后返回给Ognl.getValue
方法来处理。
经过上面的源码分析后,可以判断if
中的内容主要是通过Ognl
表达式来决定,所以我们就来验证Ognl
表达式对于"type != null and type != ''"
这个内容的判断。
OGNL表达式测试
java
public class Main {
public static void main(String[] args) throws OgnlException {
System.out.println(Ognl.parseExpression("0 != null and 0 != ''"));
}
}
可以看到"0 != null and 0 != ''"
先是被解析为了(0 != null) && (0 != "")
这样一段内容,接着我们放入Ognl.getValue
方法中再来看看结果。
java
public class Main {
public static void main(String[] args) throws OgnlException {
Object value = Ognl.getValue(Ognl.parseExpression("0 != null and 0 != ''"), null);
System.out.println(value);
}
}
结果为false
我们可以继续再对其他类型进行测试。
java
public class Main {
public static void main(String[] args) throws OgnlException {
Object value1 = Ognl.getValue(Ognl.parseExpression("1 != null and 1 != ''"), null);
Object value2 = Ognl.getValue(Ognl.parseExpression("2 != null and 2 != ''"), null);
Object value3 = Ognl.getValue(Ognl.parseExpression("-1 != null and -1 != ''"), null);
System.out.println(value1);
System.out.println(value2);
System.out.println(value3);
}
}
结果都为true
,也就是说应该只有为0
时结果才为false
。
接下来,我们将表达式部分拆开分别进行实验。
先来测试"type != null"
java
public class Main {
public static void main(String[] args) throws OgnlException {
Object value1 = Ognl.getValue(Ognl.parseExpression("1 != null"), null);
Object value2 = Ognl.getValue(Ognl.parseExpression("2 != null"), null);
Object value3 = Ognl.getValue(Ognl.parseExpression("-1 != null"), null);
Object value4 = Ognl.getValue(Ognl.parseExpression("0 != null"), null);
System.out.println(value1);
System.out.println(value2);
System.out.println(value3);
System.out.println(value4);
}
}
再来测试"type != ''"
java
public class Main {
public static void main(String[] args) throws OgnlException {
Object value1 = Ognl.getValue(Ognl.parseExpression("1 != ''"), null);
Object value2 = Ognl.getValue(Ognl.parseExpression("2 != ''"), null);
Object value3 = Ognl.getValue(Ognl.parseExpression("-1 != ''"), null);
Object value4 = Ognl.getValue(Ognl.parseExpression("0 != ''"), null);
System.out.println(value1);
System.out.println(value2);
System.out.println(value3);
System.out.println(value4);
}
}
测试后表明,错误的结果应该就是由"type != ''"
判断造成的。
查询Ognl官方文档验证
文档中已经说明,当if
中的对象为Number
类型时,将使用双精度浮点类型与0
比较,如果等于0
则视为false
,否则视为true
。
解决方式
通过前面的测试验证和官方文档说明可以看出,主要问题就是Number
类型的判断,我们只需要改写成如下这样就可以了。
xml
<if test="type != null">
</if>
不过,如果type
为数值类型,本就不应该在让它去和空字符串进行比较,这在Java语言中编译就无法通过。