当前位置: 首页 > ops >正文

MyBatis延迟加载(Lazy Loading)之“关联查询”深度解析与实践

引言

延迟加载是MyBatis优化性能的核心技术之一,特别适用于处理对象关联关系。当主实体关联的子实体数据量较大或访问频率较低时,延迟加载能显著减少不必要的数据库查询,提升系统性能。


完整代码实现

1. XML映射配置
<!-- 学生映射配置 -->
<resultMap id="studentMap" type="com.test.entity.Student"><id property="id" column="id"/><result property="name" column="name"/><association property="clazz" javaType="com.test.entity.Class"column="cid"select="com.test.repository.ClassRepository.findByClaId"/>
</resultMap><select id="findByStuId" parameterType="java.lang.Integer" resultMap="studentMap">SELECT * FROM student WHERE id = #{id}
</select><!-- 班级映射配置 -->
<select id="findByClaId" parameterType="java.lang.Integer" resultType="com.test.entity.Class">SELECT * FROM class WHERE id = #{id}
</select>
2. 仓库接口定义
// 学生仓库接口
public interface StudentRepository {Student findByStuId(Integer id);
}// 班级仓库接口
public interface ClassRepository {Class findByClaId(Integer id);
}

逐行解析配置

<resultMap id="studentMap" type="com.test.entity.Student">
  • <resultMap>:定义ORM映射规则
  • id="studentMap":映射规则唯一标识符
  • type="com.test.entity.Student":目标实体类全路径
<id property="id" column="id"/>
<result property="name" column="name"/>
  • <id>:主键字段映射(数据库id列 → 实体id属性)
  • <result>:普通字段映射(数据库name列 → 实体name属性)
<association property="clazz" javaType="com.test.entity.Class"column="cid"select="com.test.repository.ClassRepository.findByClaId"/>
  • property="clazz":映射到Student实体的clazz属性
  • javaType="com.test.entity.Class":关联属性的完整类型
  • column="cid":传递到关联查询的参数列(当前查询的cid字段值)
  • select="...":指定延迟加载的查询方法(关键配置)
<select id="findByStuId" resultMap="studentMap">SELECT * FROM student WHERE id = #{id}
</select>
  • 主查询:根据学生ID获取基本信息(不包含班级数据)

延迟加载执行流程

在这里插入图片描述


延迟加载核心特性

  1. 按需加载机制

    • 初始查询仅获取学生基础数据(不含关联对象)
    • 班级数据在首次调用getClazz()时动态加载
    • 避免不必要的数据传输和内存占用
  2. N+1查询模式

    • 1次主查询获取学生列表
    • N次关联查询获取每个学生的班级数据
    • 示例:查询10个学生 → 1(主查询)+10(班级查询)
  3. 关联查询分离

    • 主查询与关联查询完全解耦
    • 各查询可独立优化和复用
    • 通过select属性指定关联加载方法

延迟加载 vs JOIN查询

维度延迟加载JOIN查询
查询次数1 + N(按需加载)1次(复杂JOIN)
数据量首次响应快,传输量小单次传输量大
内存占用初始内存占用低一次性加载所有关联数据
适用场景关联数据访问率<30%需要立即使用所有关联数据
性能瓶颈N+1问题(批量操作时)JOIN复杂度
代码维护逻辑清晰,关联配置解耦SQL复杂度高

最佳实践与优化策略

1. 全局配置启用(mybatis-config.xml)
<settings><!-- 启用延迟加载 --><setting name="lazyLoadingEnabled" value="true"/><!-- 禁用激进加载 --><setting name="aggressiveLazyLoading" value="false"/><!-- 按需加载触发方法 --><setting name="lazyLoadTriggerMethods" value="equals,clone"/>
</settings>
2. 批量加载优化
<association property="clazz" select="com.test.repository.ClassRepository.findById"fetchType="lazy"  <!-- 显式声明加载方式 -->column="cid"/>
3. 解决N+1问题
  • 批量预加载:通过@Fetch注解配置批量加载
    @Fetch(FetchMode.SUBSELECT)
    private Class clazz;
    
  • 智能判断:根据业务场景混合使用JOIN和延迟加载
  • 二级缓存:对频繁访问的关联实体启用缓存
4. 方法命名规范

如示例所示,采用findByStuId/findByClaId的明确命名:

  • 清晰区分主查询和关联查询
  • 避免方法重载导致的歧义
  • 提高代码可读性和维护性

典型应用场景

  1. 列表-详情页模式

    • 列表页:仅加载学生基本信息(findByStuId
    • 详情页:点击时加载班级数据(触发getClazz()
  2. 大对象关联

    • 学生关联班级简历(大文本字段)
    • 初始不加载,需要时再获取
  3. 多级嵌套关联

    学生
    班级
    班主任
    教研组

    逐级延迟加载避免深度JOIN

  4. 微服务架构

    • 学生服务和班级服务分离时
    • 通过延迟加载实现跨服务数据聚合

总结

MyBatis延迟加载通过精妙的<association>配置和按需加载机制,在保证功能完整性的同时优化了系统性能。关键点在于:

  1. 使用select属性分离关联查询
  2. 通过明确的方法命名(如findByXxxId)增强可读性
  3. 合理配置全局延迟加载策略
  4. 根据业务场景选择加载策略(延迟加载/JOIN/批量加载)

正确使用延迟加载可使应用在数据层获得10x性能提升,特别适用于大型企业级应用和高并发场景。

http://www.xdnf.cn/news/15541.html

相关文章:

  • Spring原理揭秘--Spring的AOP
  • linux_线程同步
  • 「Java案例」递归实现整数的倒置
  • springboot打包二次压缩Excel导致损坏
  • git@github.com: Permission denied (publickey).
  • 基于5G系统的打孔LDPC编码和均匀量化NMS译码算法matlab性能仿真
  • 前端-HTML
  • 算法竞赛备赛——【图论】求最短路径——Dijkstra
  • 【Bluedroid】btif_a2dp_sink_init 全流程源码解析
  • (转)Kubernetes基础介绍
  • flask request实现两台PC之间文件传输通信
  • 解锁 iOS 按键精灵辅助工具自动化新可能:iOSElement.Click 让元素交互更简单
  • 【AI News | 20250716】每日AI进展
  • 《C++初阶之STL》【auto关键字 + 范围for循环 + 迭代器】
  • Java双冒号操作符全面解析
  • 【面板数据】企业劳动收入份额数据集-含代码及原始数据(2007-2022年)
  • 数据驱动视角下的黄金异动解析:多因子模型验证鲍威尔去留与金价关联性
  • PyTorch笔记8----------卷积神经网络
  • 013【入门】队列和栈-链表、数组实现
  • TCP 拥塞控制算法 —— 慢启动(Slow Start)笔记
  • 【add vs commit】Git 中的 add 和 commit 之间的区别
  • HTTP协议介绍
  • 堆排序算法详解:原理、实现与C语言代码
  • Opencv---cv::minMaxLoc函数
  • 激活函数LeakyReLU
  • ai 编程工具,简单总结
  • C++设计模式之创建型模式
  • WSL2更新后Ubuntu 24.04打不开(终端卡住,没有输出)
  • Java对象的比较
  • 每日算法刷题Day49:7.16:leetcode 差分5道题,用时2h