MyBatis如何处理懒加载和预加载?

大家好,我是锋哥。今天分享关于【MyBatis如何处理懒加载和预加载?】**面试题。**希望对大家有帮助;

MyBatis如何处理懒加载和预加载?

超硬核AI学习资料,现在永久免费了!

在 MyBatis 中,懒加载(Lazy Loading)和预加载(Eager Loading)是两种不同的查询策略,用于控制关联对象的加载时机和方式。具体处理方式如下:

1. 懒加载(Lazy Loading)

懒加载是指在查询时,只加载需要的主对象,关联的其他对象则在需要时才被加载。懒加载的优点是能减少数据库的查询次数和内存的使用,但可能会导致 N+1 查询问题。

如何启用懒加载:

  • 默认情况下,MyBatis 支持懒加载。如果需要启用懒加载,需要在 MyBatis 的配置文件中启用懒加载功能:

    <settings> <setting name="lazyLoadingEnabled" value="true"/> <setting name="aggressiveLazyLoading" value="false"/> </settings>
  • lazyLoadingEnabled: 启用懒加载。默认为 false,设为 true 时启用懒加载。

  • aggressiveLazyLoading: 默认情况下,如果访问延迟加载的属性,MyBatis 会立即执行 SQL 查询。如果该属性设为 false,则只有在实际访问时才加载关联的对象,避免了无意义的查询。

懒加载的使用: 懒加载通常用在 resultMapassociationcollection 等关联映射中,例如:

复制代码
<resultMap id="userMap" type="User">
    <id property="id" column="id"/>
    <result property="name" column="name"/>
    <association property="address" column="address_id" javaType="Address" select="selectAddressById"/>
</resultMap>

在上面的例子中,address 字段是一个懒加载属性,它会在第一次访问时才会通过 selectAddressById 查询。

2. 预加载(Eager Loading)

预加载是指在查询时,关联对象会和主对象一并加载,通常在数据库查询时一次性将关联的所有数据加载出来。

如何启用预加载:

  • 默认情况下,MyBatis 的关联对象是采用懒加载策略的。如果想要使用预加载,可以通过 fetchType 属性设置为 EAGER

    <association property="address" column="address_id" javaType="Address" fetchType="EAGER"/>
  • fetchType 可以设置为 LAZYEAGER,默认值为 LAZY。设置为 EAGER 时,关联的对象会被立即加载,而不是延迟加载。

预加载的使用:

复制代码
<resultMap id="userMap" type="User">
    <id property="id" column="id"/>
    <result property="name" column="name"/>
    <association property="address" column="address_id" javaType="Address" fetchType="EAGER"/>
</resultMap>

在这个例子中,address 字段是采用预加载策略,在查询 User 时,Address 会立即被加载出来。

总结

  • 懒加载 :关联的对象不会立即加载,只有在访问该对象时才会执行 SQL 查询。这有助于提高性能,但可能导致 N+1 查询问题。需要通过 lazyLoadingEnabledfetchType 设置来控制。
  • 预加载 :在查询主对象时,所有关联的对象会一并加载,通常会带来性能损失,因为需要加载更多数据,但避免了懒加载可能导致的额外查询。可以通过 fetchType 设置为 EAGER 来启用。

在使用时,可以根据具体场景决定是否启用懒加载或预加载,尤其在设计复杂的关联查询时,需要权衡性能与查询次数。

相关推荐
axng pmje28 分钟前
Java语法进阶
java·开发语言·jvm
rKWP8gKv740 分钟前
Java微服务性能监控:Prometheus与Grafana集成方案
java·微服务·prometheus
老前端的功夫42 分钟前
【Java从入门到入土】28:Stream API:告别for循环的新时代
java·开发语言·python
qq_4352879243 分钟前
第9章 夸父逐日与后羿射日:死循环与进程终止?十个太阳同时值班的并行冲突
java·开发语言·git·死循环·进程终止·并行冲突·夸父逐日
小江的记录本1 小时前
【Kafka核心】架构模型:Producer、Broker、Consumer、Consumer Group、Topic、Partition、Replica
java·数据库·分布式·后端·搜索引擎·架构·kafka
止语Lab1 小时前
从手动到框架:Go DI 演进的三个拐点
开发语言·后端·golang
yaoxin5211231 小时前
397. Java 文件操作基础 - 创建常规文件与临时文件
java·开发语言·python
小短腿的代码世界1 小时前
Qt日志系统深度解析:从qDebug到企业级日志框架
开发语言·qt
REDcker2 小时前
浏览器端Web程序性能分析与优化实战 DevTools指标与工程清单
开发语言·前端·javascript·vue·ecmascript·php·js
极客先躯3 小时前
高级java每日一道面试题-2025年11月24日-容器与虚拟化题[Dockerj]-runc 的作用是什么?
java·oci 的命令行工具·最小可用·无守护进程·完全标准·创建容器的核心流程·runc 核心职责思维导图