Mybatis查询时的延迟加载解析

 更新时间:2023年10月27日 11:10:58   作者:Forrit  
这篇文章主要介绍了Mybatis查询时的延迟加载解析,先从单表查询,需要时再从关联表去关联查询,能大大提高数据库性能,因为查询单表要比关联查询多张表速度要快,延迟加载分为两种:深度延时加载,侵入式延迟加载,需要的朋友可以参考下

Mybatis延迟加载

什么是延迟加载呢?

通俗的讲就是按需加载,我们需要什么的时候再去进行什么操作。

什么时候会用到延迟加载呢?

多表单独查询的时候。

而且先从单表查询,需要时再从关联表去关联查询,能大大提高数据库性能,因为查询单表要比关联查询多张表速度要快。 延迟加载分为两种:深度延时加载,侵入式延迟加载

如何开启延时加载?

两种方法:

1.在collection标签中:

  <select id="selectByMinorId"  resultMap="selectMinorMap">
        select cid,cname from country where cid =#{id}
  </select>
    
    <resultMap id="selectMinorMap" type="Country">
        <id column="cid" property="cid"/>
        <result column="cname" property="cname"/>
        <collection column="cid" property="mino" ofType="Minor" select="selectByMinorIds" fetchType="lazy"/>
				//使用fetch属性的值为true,默认开启深度延迟加载
    </resultMap>
    
    <select id="selectByMinorIds" resultType="com.example.mytest01.pojo.Minor" >
    	select mname from minor where countryid =#{cid}
	</select>

2.mybatis.xml中进行配置

<settings>

<!--延迟加载总开关,打开后,默认的是深度延时加载 -->
<setting name="lazyLoadingEnabled" value="true" />

<!--侵入式延迟加载开关 -->
<!--3.4.1版本之前默认是true,之后默认是false -->
<setting name="aggressiveLazyLoading" value="true" />

</settings>

加载特点

侵入式延迟: 执行对主加载对象的查询时,不会执行对关联对象的查询。但当要访问主加载对象的详情属性时,就会马上执行关联对象的select查询。

深度延迟: 执行对主加载对象的查询时,不会执行对关联对象的查询。访问主加载对象的详情时也不会执行关联对象的select查询。只有当真正访问关联对象的详情时,才会执行对关联对象的 select 查询。

MyBatis 的延迟加载只是对关联对象的查询有迟延设置,对于主加载对象都是直接执行查询语句的。

举例: mapper.xml:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!-- namespace:填写映射当前的Mapper接口,所有的增删改查的参数和返回值类型,
		就可以直接填写缩写,不区分大小写,直接通过方法名去找类型-->
<mapper namespace="com.example.mytest01.dao.IMinorDao">


    <select id="selectByMinorId"  resultMap="selectMinorMap">
        select cid,cname from country where cid =#{id}
    </select>
    <resultMap id="selectMinorMap" type="Country">
        <id column="cid" property="cid"/>
        <result column="cname" property="cname"/>
        <collection column="cid" property="mino" ofType="Minor" select="selectByMinorIds"/>

    </resultMap>
    <select id="selectByMinorIds" resultType="com.example.mytest01.pojo.Minor" >
    select mname from minor where countryid =#{cid}
</select>
</mapper>

当不开启延迟加载时:

@Autowired
private IMinorDao m;
 @Test
    void hah(){
      Country c =m.selectByMinorId(1);
      //System.out.println(c.getCname());
  }

在这里插入图片描述

进行两次查询;

@Autowired
private IMinorDao m;
 @Test
    void hah(){
      //Country c =m.selectByMinorId(1);
      System.out.println(c.getCname());
  }

在这里插入图片描述

进行两次查询;

重点来了: 当开深度启延迟加载时:

@Autowired
private IMinorDao m;
 @Test
    void hah(){
      Country c =m.selectByMinorId(1);
      //System.out.println(c.getCname());
  }

在这里插入图片描述

通过日志观察到,只查询一次,还未进行第二次查询;

@Autowired
private IMinorDao m;
 @Test
    void hah(){
      //Country c =m.selectByMinorId(1);
      System.out.println(c.getCname());
  }

在这里插入图片描述

当查询主加载对象的详情属性时,依旧只进行了第一次的查询; .

@Autowired
private IMinorDao m;
 @Test
    void hah(){
      //Country c =m.selectByMinorId(1);
      //System.out.println(c.getCname());
      System.out.println(c.getMino());
  }

在这里插入图片描述

看到这里我们会发现:只有当查询主加载对象的关联属性时,才进行了两次查询

当开启侵入式延迟加载时:

@Autowired
private IMinorDao m;
 @Test
    void hah(){
      Country c =m.selectByMinorId(1);
      //System.out.println(c.getCname());
  }

在这里插入图片描述

通过日志观察到,只查询一次,还未进行第二次查询;

@Autowired
private IMinorDao m;
 @Test
    void hah(){
      //Country c =m.selectByMinorId(1);
      System.out.println(c.getCname());
  }

在这里插入图片描述

与深度延迟加载不同的时:当查询主属性的详细信息时,Mybatis就已经进行了第二次的关联属性的查询;

到此这篇关于Mybatis查询时的延迟加载解析的文章就介绍到这了,更多相关Mybatis延迟加载内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 一次Jvm old过高的排查过程实战记录

    一次Jvm old过高的排查过程实战记录

    这篇文章主要给大家介绍了一次Jvm old过高的排查过程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-11-11
  • Java中的TreeSet源码解读

    Java中的TreeSet源码解读

    这篇文章主要介绍了Java中的TreeSet源码解读,TreeSet 是一个 有序集合,它扩展了 AbstractSet 类并实现了 NavigableSet 接口,对象根据其自然顺序以升序排序和存储,该 TreeSet 中使用 平衡树,更具体的一个 红黑树,需要的朋友可以参考下
    2023-09-09
  • log4j2日志异步打印(实例讲解)

    log4j2日志异步打印(实例讲解)

    下面小编就为大家带来一篇log4j2日志异步打印(实例讲解)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • mybatis中的if-else及if嵌套使用方式

    mybatis中的if-else及if嵌套使用方式

    这篇文章主要介绍了mybatis中的if-else及if嵌套使用方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • Spring Boot学习入门之统一异常处理详解

    Spring Boot学习入门之统一异常处理详解

    我们在做Web应用的时候,请求处理过程中发生错误是非常常见的情况。下面这篇文章主要给大家介绍了关于Spring Boot学习入门之统一异常处理的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下。
    2017-09-09
  • 基于java构造方法Vector删除元素源码分析

    基于java构造方法Vector删除元素源码分析

    这篇文章主要介绍了基于java构造方法中对Vector删除元素的源码分析,有需要的朋友可以借鉴参考下,希望可以有所帮助,祝大家早日升职加薪
    2021-09-09
  • 浅谈Java的虚拟机结构以及虚拟机内存的优化

    浅谈Java的虚拟机结构以及虚拟机内存的优化

    这篇文章主要介绍了Java的虚拟机结构以及虚拟机内存的优化,讲到了JVM的堆和栈空间及GC垃圾回收等重要知识,需要的朋友可以参考下
    2016-03-03
  • springboot如何获取yml里面的属性值

    springboot如何获取yml里面的属性值

    这篇文章主要介绍了springboot如何获取yml里面的属性值,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-02-02
  • 关于Java中BeanMap进行对象与Map的相互转换问题

    关于Java中BeanMap进行对象与Map的相互转换问题

    这篇文章主要介绍了利用BeanMap进行对象与Map的相互转换,通过net.sf.cglib.beans.BeanMap类中的方法来转换,效率极高,本文给大家分享实现代码,感兴趣的朋友一起看看吧
    2022-03-03
  • Java 死锁解决方案顺序锁和轮询锁

    Java 死锁解决方案顺序锁和轮询锁

    这篇文章主要介绍了Java 死锁解决方案顺序锁和轮询锁,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-05-05

最新评论