Mybatis延迟加载使用及原理

MyBatis的延迟加载是指当需要访问一个对象的关联对象时,并不是在查询主对象的同时就加载这些关联对象,而是在实际使用到关联对象时才去查询加载。这样做的好处是可以提升查询的性能,特别是对于关联关系较为复杂,数据量较大的情况。

使用延迟加载

在MyBatis配置文件中可以设置延迟加载的属性:

xml 复制代码
<settings>
  <!-- 启用延迟加载 -->
  <setting name="lazyLoadingEnabled" value="true"/>
  <!-- 当使用到关联对象的某个属性时才加载该对象 -->
  <setting name="aggressiveLazyLoading" value="false"/>
</settings>

在映射文件中,可以通过fetchType="lazy"来指定延迟加载:

xml 复制代码
<!-- 用户和订单的一对多关系 -->
<resultMap id="UserResultMap" type="User">
  <id property="id" column="id"/>
  <result property="name" column="name"/>
  <collection property="orders" column="user_id" select="selectOrders" fetchType="lazy"/>
</resultMap>

<select id="selectUser" resultMap="UserResultMap">
  SELECT * FROM user WHERE id = #{id}
</select>

<select id="selectOrders" resultMap="OrderResultMap">
  SELECT * FROM order WHERE user_id = #{userId}
</select>

延迟加载的原理

  1. 代理对象创建:MyBatis对于需要延迟加载的对象会创建一个动态代理,在你访问这个代理对象的时候,实际上会触发代理逻辑。

  2. 代理逻辑触发 :当你访问这个代理对象的某个方法时,比如getOrders(),动态代理会拦截这个调用,检查该对象的orders属性是否已经加载。

  3. SQL查询执行 :如果orders属性未加载,代理逻辑会使用之前保存的参数信息,执行对应的SQL语句来查询数据,然后将结果设置到代理对象的orders属性中。

  4. 属性设置与返回 :结果被设置之后,就可以像访问普通对象一样访问orders属性了,从此以后,对这个属性的访问不会再触发SQL查询。

源码分析

在MyBatis的源码中,延迟加载的实现涉及以下几个核心组件:

  • SqlSession:数据库会话,执行持久化操作。
  • Executor:执行器,负责SQL语句的生成和查询缓存的维护。
  • Configuration:配置信息,包含了是否启用延迟加载的设置。
  • ResultLoader:结果加载器,负责延迟加载逻辑。

延迟加载主要通过ResultLoaderloadResult方法实现:

java 复制代码
public class ResultLoader {
  // ...省略其他代码...
  
  public Object loadResult() throws SQLException {
    // 具体的查询逻辑
    List<Object> list = selectList(statement, parameter, rowBounds, executor);
    Object value = valueHandler.getResult(list);
    if (value != null && configuration.isLazyLoadingEnabled()) {
      // 如果配置了懒加载,设置已加载标志
      lazyLoader.loaded(property);
    }
    return value;
  }
}

当访问一个代理对象的方法时,MyBatis会检查是否已经加载了相应的属性。如果没有,就会调用ResultLoader.loadResult()方法来执行实际的SQL查询,并将结果加载到对象中。

代码演示

假设有一个User类和Order类,并且在User类中有一个List<Order>类型的orders属性。当访问user.getOrders()时,如果orders尚未加载,MyBatis会执行对应的查询语句,并返回订单列表。

java 复制代码
public class User {
  private Integer id;
  private String name;
  // 此属性通过MyBatis懒加载
  private List<Order> orders;

  // getters and setters...
}

细节详尽

延迟加载的实现细节主要关注以下几点:

  • 代理方式:MyBatis可以使用JDK动态代理或CGLIB来创建代理对象。
  • 触发条件:延迟加载的触发条件是访问代理对象的属性或方法。
  • 会话管理 :延迟加载发生时,必须保证原始的SqlSession还是开放状态,否则无法执行查询。
  • 线程安全:MyBatis的延迟加载不是线程安全的。两个线程同时访问同一个代理对象的同一个延迟加载属性可能会导致查询执行两次。

延迟加载为性能优化提供一种手段,但也增加了应用的复杂性。在使用时,应当权衡延迟加载带来的好处与可能出现的问题。

相关推荐
無量3 分钟前
AQS抽象队列同步器原理与应用
后端
9号达人39 分钟前
支付成功订单却没了?MyBatis连接池的坑我踩了
java·后端·面试
用户497357337981 小时前
【轻松掌握通信协议】C#的通信过程与协议实操 | 2024全新
后端
草莓熊Lotso1 小时前
C++11 核心精髓:类新功能、lambda与包装器实战
开发语言·c++·人工智能·经验分享·后端·nginx·asp.net
追逐时光者1 小时前
精选 8 个 .NET 开发实用的类库,效率提升利器!
后端·.net
a程序小傲2 小时前
京东Java面试被问:Fork/Join框架的使用场景
java·开发语言·后端·postgresql·面试·职场和发展
想用offer打牌2 小时前
面试官问Redis主从延迟导致脏数据读怎么解决?
redis·后端·面试
appearappear2 小时前
Mac 上重新安装了Cursor 2.2.30,重新配置 springboot 过程记录
java·spring boot·后端
谷哥的小弟3 小时前
Spring Framework源码解析——RequestContext
java·后端·spring·框架·源码
IT_陈寒3 小时前
Vite 5大优化技巧:让你的构建速度飙升50%,开发者都在偷偷用!
前端·人工智能·后端