在上一篇博客​​Mybatis源码分析--关联表查询及延迟加载(一)​​中我们简单介绍了Mybatis的延迟加载的编程,接下来我们通过分析源码来分析一下Mybatis延迟加载的实现原理。

其实简单来说Mybatis的延迟加载就是分多次执行SQL语句,这样就实现了延迟加载的机制,并且第一次执行的结果值可能是接下来执行的SQL语句的参数值,Mybatis实现执行接下来的SQL的原理机制是通过代理类来实现的,就是第一次执行的结果对象其实已经是一个代理对象,当执行接下来相关的对象时会执行其他SQL语句,这样就实现了延迟加载的机制。

其实现在DefaultResultSetHandler的createResultObject中。

private Object createResultObject(ResultSetWrapper rsw, ResultMap resultMap, ResultLoaderMap lazyLoader, String columnPrefix) throws SQLException {
final List<Class<?>> constructorArgTypes = new ArrayList<Class<?>>();
final List<Object> constructorArgs = new ArrayList<Object>();
final Object resultObject = createResultObject(rsw, resultMap, constructorArgTypes, constructorArgs, columnPrefix);
if (resultObject != null && !hasTypeHandlerForResultObject(rsw, resultMap.getType())) {
final List<ResultMapping> propertyMappings = resultMap.getPropertyResultMappings();
for (ResultMapping propertyMapping : propertyMappings) {
// issue gcode #109 && issue #149
//如果配置了延迟加载,这是返回的对象为代理对象
if (propertyMapping.getNestedQueryId() != null && propertyMapping.isLazy()) {
return configuration.getProxyFactory().createProxy(resultObject, lazyLoader, configuration, objectFactory, constructorArgTypes, constructorArgs);
}
}
}
return resultObject;
}


mybatis提供了两种方式来实现代理机制configuration.getProxyFactory().createProxy(resultObject, lazyLoader, configuration, objectFactory, constructorArgTypes, constructorArgs)最终调用的是如下两个实现类,目的就是创建代理类。

Mybatis源码分析--关联表查询及延迟加载原理(二)_sql语句

其实这样的结果就是结果对象teacher就是一个代理类,当执行和User相关的操作时会执行User相关的SQL语句,这样就简单的实现了延迟加载机制。

Teacher teacher =  (Teacher) session.selectList(statement, 1).get(0);
ystem.out.println(teacher.getName());
List<User> users = teacher.getUsers();
System.err.println(users.get(3).getAge());

当我们执行如下语句

System.err.println(teacher.getClass());

获得的结果是:可以看出teacher已经是一个代理类了

class com.tianjunwei.lazy.entity.Teacher_$$_jvstb9_0。

Mybatis源码分析--关联表查询及延迟加载原理(二)_代理类_02