MyBatis如何处理延迟加载?

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

MyBatis如何处理延迟加载?

1000道 互联网大厂Java工程师 精选面试题-Java资源分享网

MyBatis 支持 延迟加载(Lazy Loading),允许在需要数据时才从数据库加载,而不是在查询结果第一次返回时就立即加载所有数据。这种方式可以优化性能,减少不必要的数据库查询,提高应用的响应速度和资源利用效率。

延迟加载的原理

延迟加载的核心思想是,将关联对象或集合的加载推迟到真正需要时才进行加载,而不是在主查询时一次性加载。这可以通过两种主要方式实现:

  • 延迟加载单个关联对象(如:查询时只加载主对象,关联对象在访问时才加载)
  • 延迟加载集合属性(如:查询时不加载集合,只有在访问集合时才进行查询)

MyBatis 通过代理模式(代理对象)和懒加载机制来实现延迟加载。以下是 MyBatis 如何处理延迟加载的详细信息:

1. 开启延迟加载

MyBatis 默认开启延迟加载机制,但你需要在配置文件中指定相关配置。

在 mybatis-config.xml 配置文件中:
<settings>
  <setting name="lazyLoadingEnabled" value="true"/>
  <setting name="aggressiveLazyLoading" value="false"/>
  <setting name="lazyLoadTriggerMethods" value="equals,clone,hashCode"/>
</settings>
  • lazyLoadingEnabled:开启延迟加载,默认为 true,启用后,MyBatis 会对关联的对象进行延迟加载。
  • aggressiveLazyLoading:是否强制加载所有的延迟加载属性。默认为 false,即延迟加载会在实际访问时才加载。
  • lazyLoadTriggerMethods:定义了触发延迟加载的 Java 方法,通常是 equalshashCode 或 clone 等方法。当调用这些方法时,MyBatis 会检查是否需要延迟加载关联对象。

2. 延迟加载的配置

在 MyBatis 中,可以通过以下几种方式配置延迟加载:

2.1 使用 resultMap 配置延迟加载

延迟加载通常与映射关系中的关联对象一起使用。例如,当你在 resultMap 中映射一个对象时,可以通过 fetchType 来指定加载策略。

<resultMap id="orderResultMap" type="Order">
    <id property="id" column="order_id"/>
    <result property="user" column="user_id" fetchType="lazy"/>
</resultMap>
  • fetchType="lazy":表示该属性(关联对象)采用延迟加载,只有在访问时才会从数据库加载。
  • fetchType="eager":表示该属性(关联对象)采用立即加载,查询时就会一起加载。

fetchType@Many@One 注解的属性,控制关联对象的加载方式。

2.2 使用 @One 和 @Many 注解进行延迟加载

如果使用注解方式进行映射,可以使用 @One@Many 注解来指定加载策略:

public class Order {
    private int id;
    private String name;

    @One(fetchType = FetchType.LAZY)
    private User user; // 延迟加载 User 对象
}

在这种配置下,user 关联对象会在访问时触发延迟加载。

2.3 在查询时设置延迟加载

在某些情况下,你可能希望通过在查询方法中控制延迟加载。你可以在 Mapper 中使用 @Select 注解来控制:

@Select("SELECT * FROM orders WHERE id = #{id}")
@Results({
    @Result(property = "user", column = "user_id", one = @One(select = "com.example.mapper.UserMapper.selectUser", fetchType = FetchType.LAZY))
})
Order selectOrderById(int id);

此时 user 关联对象会被延迟加载。

3. 如何触发延迟加载

延迟加载是通过代理对象实现的。当你访问被延迟加载的关联对象时,MyBatis 会触发数据库查询。

例如,假设有一个 Order 对象,它有一个关联的 User 对象,在访问 Order 对象时,User 不会立即加载,只有当你访问 Order.getUser() 时,MyBatis 才会执行 SQL 查询,加载 User 数据。

Order order = orderMapper.selectOrderById(1);
User user = order.getUser();  // 此时会触发延迟加载,查询 User 数据

4. 延迟加载的代理机制

为了支持延迟加载,MyBatis 在内部使用 代理模式 来创建一个代理对象。当你访问延迟加载的属性时,代理对象会触发实际的数据库查询,并将查询结果赋给原始对象。代理对象会在数据库查询后进行填充,并返回给调用者。

  • JDK 动态代理:当延迟加载的对象实现了接口时,MyBatis 会使用 JDK 动态代理来延迟加载。
  • CGLIB 动态代理:当延迟加载的对象没有实现接口时,MyBatis 会使用 CGLIB 动态代理来实现。

5. 注意事项

  • 性能问题:虽然延迟加载有助于减少不必要的数据库查询,但过度使用延迟加载也可能导致 N+1 查询问题。例如,如果你有一个包含多个订单的列表,每个订单的用户都是延迟加载的,那么可能会触发多次数据库查询(一次查询订单表,之后每个订单查询一次用户表)。可以通过 <fetchType="eager"><join fetch> 等优化策略来避免此问题。

  • 事务问题:延迟加载通常需要在同一个事务范围内进行。如果在关闭事务后访问延迟加载的属性,可能会抛出 LazyInitializationException 异常。为了避免这种问题,可以使用 OpenSessionInView 模式,确保事务在视图渲染期间保持开启。

总结

MyBatis 提供了强大的延迟加载功能,它通过代理模式、fetchType 配置以及延迟加载触发机制来实现数据的按需加载。配置合理的延迟加载能够优化性能,减少不必要的数据库查询,但也需要小心处理 N+1 查询问题事务管理

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/944487.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Quartz任务调度框架实现任务动态执行

说明&#xff1a;之前使用Quartz&#xff0c;都是写好Job&#xff0c;指定一个时间点&#xff0c;到点执行。最近有个需求&#xff0c;需要根据前端用户设置的时间点去执行&#xff0c;也就是说任务执行的时间点是动态变化的。本文介绍如何用Quartz任务调度框架实现任务动态执行…

vue3使用video-player实现视频播放(可拖动视频窗口、调整大小)

1.安装video-player npm install video.js videojs-player/vue --save在main.js中配置全局引入 // 导入视频播放组件 import VueVideoPlayer from videojs-player/vue import video.js/dist/video-js.cssconst app createApp(App) // 视频播放组件 app.use(VueVideoPlayer)2…

从入门到精通:Vim 高效文本编辑全面指南

文章目录 前言&#x1f9ec;一、Vim 的编辑哲学&#xff1a;模式分离与高效键盘操作&#x1f9ec;二、基础命令与快捷键&#xff1a;从简单到熟悉&#x1f9ec;三、进阶功能&#xff1a;多文件、分屏与可视化模式&#x1f9ec;四、自定义配置与 .vimrc&#xff1a;打造你的专属…

正则表达式(三剑客之sed)

1.sed工具的使用 1.1 sed工具 1&#xff09;命令格式&#xff1a;sed -n ‘n’ p filename 1.2 打印某行 1&#xff09;打印第二行 [rootlocalhost ~]# sed -n 2p /etc/passwd 2&#xff09;第二行重复打印 [rootlocalhost ~]# sed 2p /etc/passwd 3&#xff09;所有行全部…

珞珈一号夜光遥感数据地理配准,栅格数据地理配准

目录 一、夜光数据下载&#xff1a; 二、夜光遥感数据地理配准 三、计算夜光数据值 四、辐射定标 五、以表格显示分区统计 五、结果验证 夜光数据位置和路网位置不匹配&#xff0c;虽然都是WGS84坐标系&#xff0c;不匹配&#xff01;&#xff01;&#xff01;不要看到就直接…

虚幻引擎是什么?

Unreal Engine&#xff0c;是一款由Epic Games开发的游戏引擎。该引擎主要是为了开发第一人称射击游戏而设计&#xff0c;但现在已经被成功地应用于开发模拟游戏、恐怖游戏、角色扮演游戏等多种不同类型的游戏。虚幻引擎除了被用于开发游戏&#xff0c;现在也用于电影的虚拟制片…

多个微服务 Mybatis 过程中出现了Invalid bound statement (not found)的特殊问题

针对多个微服务的场景&#xff0c;记录一下这个特殊问题&#xff1a; 如果启动类上用了这个MapperScan注解 在resource 目录下必须建相同的 com.demo.biz.mapper 目录结构&#xff0c;否则会加载不到XML资源文件 。 并且切记是com/demo/biz 这样的格式创建&#xff0c;不要使用…

使用Excel制作通达信自定义外部数据,安排!!!

Excel相信大家电脑上都有这个工具&#xff0c;相比敲编程代码&#xff0c;用这个去做自定义数据对大多数人&#xff0c;应该是比较友好的。自定义数据分为外部序列数据&#xff0c;看了一下内容理解起来比较多&#xff0c;分两期给大家介绍。为了照顾电脑基础薄弱的朋友&#x…

win10、win11-鼠标右键还原、暂停更新

系统优化 win 10jihuo win 11jihuo鼠标右键还原暂停更新 update 2024.12.28win 10 jihuo winx&#xff0c;打开powershell管理员&#xff0c;输入以下命令,选择1并等待 irm https://get.activated.win | iex参考&#xff1a;https://www.bilibili.com/video/BV1TN411M72J/?sp…

QT集成IntelRealSense双目摄像头2,集成OpenGL

上一篇文章写了如何把IntelRealSense摄像头的SDK集成到QT项目&#xff0c;并成功采集数据&#xff0c;在没有用OpenCV的情况下完成色彩数据&#xff0c;以及深度数据的显示。 具体地址&#xff1a;https://blog.csdn.net/qujia121qu/article/details/144734163 本次主要写如何…

数据分析的分类和EDIT思维框架

为了服务于企业不同层次的决策&#xff0c;商业数据分析过程需要提供相应的数据科学产出物。 一般而言&#xff0c;数据分析需要经历从需求层、数据层、分析层到输出层四个阶段。 第一个阶段是需求层——确定目标&#xff0c;具体目标需要依据具体的层次进行分析&#xff1a…

面试场景题系列:设计URL短链

1.场景需求界定 1.缩短URL&#xff1a;提供一个长URL&#xff0c;返回一个短很多的URL。 2.重定向URL&#xff1a;提供一个缩短了的URL&#xff0c;重定向到原URL。 3.高可用、可扩展性和容错性考量。 •写操作&#xff1a;每天生成1亿个URL。 •每秒的写操作数&#xff1a…

Linux 基本指令

目录 1.常见指令 1.1 ls指令 1.2 pwd指令 1.3 cd指令 1.4 touch指令 1.5 mkdir指令 1.6 rm和rmdir指令 1.7 man指令 1.8 cp指令 1.9 mv指令 ​编辑 1.10 cat指令 1.11 more指令 1.12 less指令 1.13 head指令 1.14.tail指令 1.15 时间相关的指令 1.16 cal…

WEB UI 创建视图

1 视图名称 (点第1创建视图) 2 模型节点 可以空 3 上下文节点 4 新增节点下的属性 &#xff0c;参考结构(先建好的结构) 5 选择视图类型&#xff1a;&#xff08;表单&#xff0c; 列表&#xff09; 表单 &#xff1a;单条数据 列表 &#xff1a;多条数据&#xff08;表格…

redis cluster实验详解

华子目录 实验环境准备部署redis cluster添加节点删除节点redis cluster集群维护 实验 环境准备 再开3台主机 先把之前3台源码编译的redis删除 [rootredis-node1 ~]# cd /usr/local/redis/ [rootredis-node1 redis]# make uninstall[rootredis-node2 ~]# cd /usr/local/redi…

【详细讲解】hive优化

1、开启本地模式 大多数的Hadoop Job是需要Hadoop提供的完整的可扩展性来处理大数据集的。不过&#xff0c;有时Hive的输入数据量是非常小的。在这种情况下&#xff0c;为查询触发执行任务消耗的时间可能会比实际job的执行时间要多的多。对于大多数这种情况&#xff0c;Hive可…

Unity3d UGUI如何优雅的实现Web框架(Vue/Rect)类似数据绑定功能(含源码)

前言 Unity3d的UGUI系统与Web前端开发中常见的数据绑定和属性绑定机制有所不同。UGUI是一个相对简单和基础的UI系统&#xff0c;并不内置像Web前端&#xff08;例如 Vue.js或React中&#xff09;那样的双向数据绑定或自动更新UI的机制。UGUI是一种比较传统的 UI 系统&#xff…

828华为云征文|使用sysbench对Flexus X实例对mysql进行性能测评

目录 一、Flexus X实例概述 1.1?Flexus X实例 1.2?在mysql方面的优势 二、在服务器上安装MySQL 2.1 在宝塔上安装docker 2.2 使用宝塔安装mysql 2.3 准备测试数据库和数据库表 三、安装sysbench并进行性能测试 3.1 使用yum命令sysbench 3.2?运行?sysbench 并进行…

影刀进阶指令 | Kimi (对标ChatGPT)

文章目录 影刀进阶指令 | Kimi &#xff08;对标ChatGPT&#xff09;一. 需求二. 流程三. 实现3.1 流程概览3.2 流程步骤讲解1\. 确定问题2\. 填写问题并发送3\. 检测答案是否出完 四. 运维 影刀进阶指令 | Kimi &#xff08;对标ChatGPT&#xff09; 简单讲讲RPA调用kimi实现…

【教程】通过Docker运行AnythingLLM

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你&#xff0c;欢迎[点赞、收藏、关注]哦~ 官方教程&#xff1a;Local Docker Installation ~ AnythingLLM 1、先创建一个目录用于保存anythingllm的持久化文件&#xff1a; sudo mkdir /app su…