mybatis 是否支持延迟加载?延迟加载的原理是什么?

server/2025/2/12 21:08:57/

1. MyBatis 是否支持延迟加载?

是的,MyBatis 支持延迟加载。延迟加载的主要功能是推迟数据加载的时机,直到真正需要时再去加载。这种方式能提高性能,尤其是在处理关系型数据时,可以避免不必要的数据库查询。

具体来说,MyBatis 支持在关联对象(association)和关联集合(collection)上进行延迟加载。通常,association 用于一对一的关联查询,collection 用于一对多的关联查询。

配置延迟加载

要启用延迟加载功能,您需要在 MyBatis 的配置文件中设置 lazyLoadingEnabledtrue

<settings><setting name="lazyLoadingEnabled" value="true"/>
</settings>

通过这种方式,MyBatis 在查询主表数据时,关联表的数据不会立即加载,而是等到访问关联属性时再去查询。

下面来详细说明一下:

举例:

假设有两个表 UserAddress,它们之间有一对一的关系,一个用户对应一个地址。

java">public class User {private Integer id;private String name;private Address address; // 延迟加载// getters and setters
}public class Address {private Integer id;private String street;// getters and setters
}

 如果在 User 类中的 address 属性上启用延迟加载,则在查询 User 时,address 这一关联数据不会立即加载,而是等到我们访问 address 时才会去查询 Address 表。

2. 表结构的关系

User 类和 Address 类代表了数据库中的两张表,User 表和 Address 表。在实际的数据库设计中,User 表和 Address 表之间通常会有外键关联,例如 Address 表可能会有一个字段 user_id,用来表示 AddressUser 的一对一关系。

假设我们的数据库表结构如下:

  • User 表

    CREATE TABLE user (id INT PRIMARY KEY,name VARCHAR(100)
    );
    
  • Address 表

    CREATE TABLE address (id INT PRIMARY KEY,street VARCHAR(100),user_id INT,  -- 关联到 User 表FOREIGN KEY (user_id) REFERENCES user(id)
    );
    

这里,Address 表中的 user_id 字段是一个外键(即使数据库表没有外键约束,MyBatis 依然可以根据配置的 SQL 查询进行延迟加载。外键约束只是保证了数据完整性,并不影响 MyBatis 如何进行延迟加载。),指向 User 表中的 id 字段,表示每个地址属于某个用户。

3. MyBatis 中如何关联表

在 MyBatis 中,User 类的 address 属性是延迟加载的,即当我们访问 user.getAddress() 时,MyBatis 会根据 user_id 来查询 Address 表中的数据。这是通过 MyBatis 的映射配置 来实现的。

MyBatis 映射配置

在 MyBatis 中,我们可以通过 映射文件(Mapper XML)注解 来指定如何加载 UserAddress 之间的关联数据。

假设我们使用的是 XML 配置,可能会在 User 类和 Address 类之间进行关联配置。

例如,定义一个查询 User 对象时,同时懒加载它的 Address 属性:

<!-- UserMapper.xml -->
<resultMap id="userResultMap" type="User"><id property="id" column="id"/><result property="name" column="name"/><!-- 延迟加载 address --><association property="address" column="id" select="selectAddressByUserId" fetchType="lazy"/>
</resultMap><select id="getUserById" resultMap="userResultMap">SELECT id, name FROM user WHERE id = #{id}
</select><!-- 查询 address -->
<select id="selectAddressByUserId" resultType="Address">SELECT * FROM address WHERE user_id = #{userId}
</select>

在这里:

  • resultMap 配置了如何将数据库查询结果映射到 User 对象。address 属性是通过关联查询来获取的(association)。
  • selectAddressByUserId 这个 SQL 查询通过 user_id 字段来查找关联的 Address 数据。

4. 延迟加载的实际流程

  • 当你查询 User 对象时(例如 SELECT id, name FROM user WHERE id = #{id}),只会查询 User 表,address 字段不会被立即加载。
  • User 对象被加载后,如果你访问 user.getAddress(),MyBatis 会检查 address 是否已经加载。如果没有加载,MyBatis 会发起一个新的查询,查询与当前 user 关联的 Address 数据。

具体来说,代理对象会拦截你对 user.getAddress() 的调用,然后执行 selectAddressByUserId 查询:

SELECT * FROM address WHERE user_id = #{userId}

#{userId} 会被替换为你查询的 User 对象的 id,然后执行查询,返回的 Address 对象会被设置到 User 对象的 address 属性中。

5. 总结:

  • 在数据库中,Address 表通过 user_id 字段与 User 表关联。
  • MyBatis 在映射文件中配置了 UserAddress 之间的关系,并且配置了延迟加载。
  • 初次查询 User 数据时,只会查询 User 表的数据,address 不会立即加载。
  • 当你访问 user.getAddress() 时,MyBatis 会触发一个新的查询(SELECT * FROM address WHERE user_id = #{userId}),加载 Address 数据并返回给你。

因此,SELECT * FROM address WHERE user_id = #{userId} 这个查询是由 MyBatis 根据 User 对象的 id 字段动态生成的,目的是通过外键 user_id 查询到与该 User 对象关联的 Address 数据。这是通过映射文件中的 association 和延迟加载配置来实现的。


http://www.ppmy.cn/server/167152.html

相关文章

ubuntu 22.04 安装 cuda sdk 11.8

ubuntu 22.04 安装 cuda sdk 11.8 linux kernel 版本太高的问题 主要思路是先安装 nv 显卡驱动&#xff0c;这会同时安装 kmd driver 然后安装 cuda sdk 11.x 时不安装 kernel driver 下载 display driver 搜索 display driver https://www.nvidia.com/en-us/drivers/ 选择比…

波导阵列天线学习笔记8 高增益、低轴比的3D打印Ka波段圆极化单脉冲天线阵列

摘要&#xff1a; 本文中&#xff0c; 一种3D打印的16x16圆极化单脉冲天线阵列在Ka波段研究&#xff0c;有着高增益和低轴比的特点。此单脉冲天线阵列有着四个低剖面的左旋圆极化子阵列和一个顺序旋转的和差网络。这四个子阵列正交连接着和差网络的输出&#xff0c;保证了传统2…

云原生周刊:DeepSeek 颠覆人工智能

开源项目推荐 Ollama Ollama 是一个开源的 AI 工具&#xff0c;旨在为用户提供简单而强大的本地部署语言模型解决方案。它支持直接在本地计算机上运行多个预训练的语言模型&#xff0c;能够提供与云端类似的体验&#xff0c;但无需依赖外部服务器或网络连接。 Ollama 的主要…

DeepSeek使用技巧大全(含本地部署教程)

在人工智能技术日新月异的今天&#xff0c;DeepSeek 作为一款极具创新性和实用性的 AI&#xff0c;在众多同类产品中崭露头角&#xff0c;凭借其卓越的性能和丰富的功能&#xff0c;吸引了大量用户的关注。 DeepSeek 是一款由国内顶尖团队研发的人工智能&#xff0c;它基于先进…

SQLMesh系列教程-2:SQLMesh入门项目实战(上篇)

假设你已经了解SQLMesh是什么&#xff0c;以及其他应用场景。如果没有&#xff0c;我建议你先阅读《SQLMesh系列教程-1&#xff1a;数据工程师的高效利器-SQLMesh》。 在本文中&#xff0c;我们将完成一个小项目或教程&#xff0c;以帮助你开始使用SQLMesh。你可以选择一步一步…

详解策略模式

引言 实现一个目标往往有多种方式&#xff0c;比如从上海到北京&#xff0c;可以选择高铁、火车、飞机、自驾等等。同样实现一个功能我们可能也有多种方法&#xff0c;把这些方法封装为算法&#xff0c;根据不同的需求选择不同的算法&#xff08;策略&#xff09;&#xff0c;让…

【人工智能】如何在VSCode中使用DeepSeek?

文章目录 前言一、准备工作二、安装DeepSeek插件步骤1、扩展图标搜索DeepSeep2、安装DeepSeek插件3、使用测试DeepSeekBito文心一言 结论 前言 介绍在VSCode中调用DeepSeek插件工具&#xff0c;可以进行对话、编码。 一、准备工作 确保已经安装好了VSCode软件。 二、安装D…

DevOps的个人学习

一、DevOps介绍 软件开发最初是由两个团队组成&#xff1a; 开发团队&#xff1a;负责设计和构建系统。运维团队&#xff1a;负责测试代码后部署上线&#xff0c;确保系统稳定安全运行。 这两个看似目标不同的团队需要协同完成一个软件的开发。DevOps整合了开发与运维团队&a…