聊聊Mybatis的动静Sql之这三个SqlNode同样重要
ForEachSqlNode
ForEachSqlNode是解析出 <foreach> 的对象,
@Override
public boolean apply(DynamicContext context) {
Map<String, Object> bindings = context.getBindings();
final Iterable<?> iterable = evaluator.evaluateIterable(collectionExpression, bindings,
Optional.ofNullable(nullable).orElseGet(configuration::isNullableOnForEach));
if (iterable == null || !iterable.iterator().hasNext()) {
return true;
}
boolean first = true;
applyOpen(context);
int i = 0;
for (Object o : iterable) {
DynamicContext oldContext = context;
if (first || separator == null) {
context = new PrefixedContext(context, "");
} else {
context = new PrefixedContext(context, separator);
}
int uniqueNumber = context.getUniqueNumber();
// Issue #709
if (o instanceof Map.Entry) {
@SuppressWarnings("unchecked")
Map.Entry<Object, Object> mapEntry = (Map.Entry<Object, Object>) o;
applyIndex(context, mapEntry.getKey(), uniqueNumber);
applyItem(context, mapEntry.getValue(), uniqueNumber);
} else {
applyIndex(context, i, uniqueNumber);
applyItem(context, o, uniqueNumber);
}
contents.apply(new FilteredDynamicContext(configuration, context, index, item, uniqueNumber));
if (first) {
first = !((PrefixedContext) context).isPrefixApplied();
}
context = oldContext;
i++;
}
applyClose(context);
context.getBindings().remove(item);
context.getBindings().remove(index);
return true;
}
解析<foreach> 的collection属性的表达式,失去汇合,增加open属性的字符串
遍历汇合,构建PrefixedContext对象,如果是Map类型,将key value保留到PrefixedContext对象中,不是的话就将汇合元素下标记和汇合元素保留到PrefixedContext对象中,调用SqlNode的apply()办法,解决#{}占位符
调用applyClose()办法增加close属性后缀
删除DynamicContext对象中的index和item
VarDeclSqlNode
VarDeclSqlNode是解析<bind>标签的类
public class VarDeclSqlNode implements SqlNode {
private final String name;
private final String expression;
public VarDeclSqlNode(String var, String exp) {
name = var;
expression = exp;
}
@Override
public boolean apply(DynamicContext context) {
final Object value = OgnlCache.getValue(expression, context.getBindings());
context.bind(name, value);
return true;
}
}
name保留了bind标签的name属性值,value保留bind标签的value属性值,apply()办法中利用OGNL解析表达式的值,而后把name和value绑定到DynamicContext中,这样就能依据name获取到value值了。
ChooseSqlNode
ChooseSqlNode是解析出choose标签的类
public class ChooseSqlNode implements SqlNode {
private final SqlNode defaultSqlNode;
private final List<SqlNode> ifSqlNodes;
public ChooseSqlNode(List<SqlNode> ifSqlNodes, SqlNode defaultSqlNode) {
this.ifSqlNodes = ifSqlNodes;
this.defaultSqlNode = defaultSqlNode;
}
@Override
public boolean apply(DynamicContext context) {
for (SqlNode sqlNode : ifSqlNodes) {
if (sqlNode.apply(context)) {
return true;
}
}
if (defaultSqlNode != null) {
defaultSqlNode.apply(context);
return true;
}
return false;
}
}
它的apply()办法也比较简单:遍历所有ifSqlNodes,也是when标签的对象信息,调用各自的apply()办法,返回胜利示意when条件成立,就返回true,defaultSqlNode不为空也是true,其余为false
总结
这篇文章次要讲了解析foreach标签的类ForEachSqlNode、解析<bind>标签的VarDeclSqlNode类和解析choose标签的ChooseSqlNode类,同时介绍这三个类的apply()办法的实现,这三个类都是SqlNode的实现类,都是对xml文件中的标签解析成的对象,这些SqlNode对象会被解析解决成SqlSource接口和它的实现类,供数据库执行,下篇文章咱们将介绍SqlSource接口和它的实现类
对文章中内容感兴趣的小伙伴能够搜寻微信公众号:敲代码的老贾,支付相应材料
发表回复