mybatis 延迟加载的深入理解

2025-05-29 0 36

什么是延迟加载

延迟加载又叫懒加载,也叫按需加载,也就是说先加载主信息,需要的时候,再去加载从信息。代码中有查询语句,当执行到查询语句时,并不是马上去db中查询,而是根据设置的延迟策略将查询向后推迟。

什么时候会执行延迟加载

配置之后在对关联对象进行查询时使用延迟加载

延迟加载策略

直接加载

遇到代码中查询语句,马上到db中执行select语句进行查询。(这种只能用于多表单独查询)

侵入式延迟加载

将关联对象的详情(具体数据,如id、name)侵入到主加载对象,作为主加载对象的详情的一部分出现。当要访问主加载对象的详情时才会查询主表,但由于关联对象详情作为主加载对象的详情一部分出现,所以这个查询不仅会查询主表,还会查询关联表。

深度延迟加载

将关联对象的详情(具体数据,如id、name)侵入到主加载对象,作为主加载对象的详情的一部分出现。当要访问主加载对象的详情时才会查询主表,但由于关联对象详情作为主加载对象的详情一部分出现,所以这个查询不仅会查询主表,还会查询关联表。

使用延迟加载的目的

减轻db服务器的压力,因为我们延迟加载只有在用到需要的数据才会执行查询操作。

如何开启延迟加载功能

mybatis延迟加载功能默认是关闭的

需要在sqlmapconfig.xml文件中通过setting标签配置来开启延迟加载功能

开启延迟加载的属性:

  • lazyloadingenabled:全局性设置懒加载。如果设为‘false',则所有相关联的都会被初始化加载。默认为false
  • aggressivelazyloading:当设置为‘true'的时候,懒加载的对象可能被任何懒属性全部加载。否则,每个属性都按需加载。默认为true

配置

?

1

2

3

4

5
<settings>

<setting name ="aggressivelazyloading" value="false"/>

<!--开启延迟加载-->

<setting name="lazyloadingenabled" value="true"/>

</settings>

我们用关联查询来实现延迟加载,假设我们现在要查出用户和用户角色。

首先我们在user中添加查询uservo的方法和xml。

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37
<!--usermapper.xml-->

....

<resultmap id="baseresultmap" type="com.redstar.basemapper.pojo.user">

<id column="id" jdbctype="varchar" property="id"/>

<result column="name" jdbctype="varchar" property="name"/>

<result column="age" jdbctype="integer" property="age"/>

<result column="role_id" jdbctype="integer" property="roleid"/>

</resultmap>

<resultmap id="userrolemapselect" type="com.redstar.basemapper.pojo.uservo">

<association property="user" resultmap="baseresultmap"/>

<association property="role" fetchtype="lazy" column="{id=role_id}"

select="com.redstar.basemapper.dao.rolemapper.getrolebyid"/>

</resultmap>

<sql id="base_column_list">

id, `name`, age, role_id

</sql>

<select id="getuservo" resultmap="userrolemapselect">

select * from user where id=#{userid}

</select>

...

<!--rolemapper.xml-->

...

<resultmap id="baseresultmap" type="com.redstar.basemapper.pojo.role">

<id column="id" jdbctype="integer" property="id" />

<result column="role_name" jdbctype="varchar" property="rolename" />

</resultmap>

<sql id="base_column_list">

id, role_name

</sql>

<select id="getrolebyid" resultmap="baseresultmap">

select * from role where id=#{id}

</select>

...

测试用例

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17
@runwith(springrunner.class)

@springboottest

public class basemapperapplicationtests {

@autowired

private usermapper usermapper;

@autowired

private rolemapper rolemapper;

@test

public void getuservo() {

system.out.println(usermapper.getuservo("12312232"));

// system.out.println(usermapper.getuserbyid("12312232"));

// system.out.println(rolemapper.getrolebyid(1));

}

}

输出结果:

uservo{user=user [hash = 1937575946, id=12312232, name=哇哈哈啊娃哈哈哇哈哈哈哈哈哈哈, age=48, roleid=1, serialversionuid=1], role=role [hash = 317053574, id=1, rolename=admin, serialversionuid=1]}

注意

许多对延迟加载原理不太熟悉的朋友会经常遇到一些莫名其妙的问题:有些时候延迟加载可以得到数据,有些时候延迟加载就会报错,为什么会出现这种情况呢?

mybatis 延迟加载是通过动态代理实现的,当调用配直为延迟加载的属性方法时, 动态代理的操作会被触发,这些额外的操作就是通过 mybatis 的 sqlsessio口去执行嵌套 sql 的 。由于在和某些框架集成时, sqlsession 的生命周期交给了框架来管理,因此当对象超出sqlsession 生命周期调用时,会由于链接关闭等问题而抛出异常 。 在和 spring 集成时,要确保只能在 service 层调用延迟加载的属性 。 当结果从 service 层返回至 controller 层时, 如果获取延迟加载的属性值,会因为 sqlsessio口已经关闭而抛出异常 。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持快网idc。

原文链接:https://segmentfault.com/a/1190000017895924

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 mybatis 延迟加载的深入理解 https://www.kuaiidc.com/109853.html

相关文章

发表评论
暂无评论