SSM-MyBatis-16:Mybatis中延迟加载

简介:   ------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥-------------     延迟加载:   所谓延迟加载是什么?   从字面意思理解到的是等一会再加载   从行为分析,他主要是缓解数据库压力,提高性能的意义   所谓延迟加载可以做到,我访问你的时候,我...

 

 

------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥-------------

 

 

延迟加载:

  所谓延迟加载是什么?

  从字面意思理解到的是等一会再加载

  从行为分析,他主要是缓解数据库压力,提高性能的意义

  所谓延迟加载可以做到,我访问你的时候,我用到你的时候,你再做加载

  延迟加载用在关联查询的多条SQL语句上

延迟加载的分类:

  直接加载------------》(默认的)这个由于是默认的,所以这个可以不做配置

  侵入式延迟加载------------》(需要配置)在执行完第一条sql时候,得到一个对象,没有访问这个对象的任何属性时,不会走第二条SQL,也就是不加载这个对象里包含的其他对象

  深度延迟加载--------》(需要配置)在执行完第一条sql时候,得到一个对象,即使访问到这个对象的属性,只要不是那个关联的对象,也不会走第二条sql,当访问这个关联的对象时,走第二条sql,

            给这个对象拿到值

下面我把实现延迟加载的方式放到下面,并做小实验,证明我上面说的话

  实体类

//Dept类
import java.util.ArrayList;
import java.util.List;

/**
 * Created by Dawn on 2018/2/26.
 */
public class Dept {
    private Integer deptNo;
    private String deptName;
    private List<Emp> emps=new ArrayList<Emp>();

    public Integer getDeptNo() {
        return deptNo;
    }

    public void setDeptNo(Integer deptNo) {
        this.deptNo = deptNo;
    }

    public String getDeptName() {
        return deptName;
    }

    public void setDeptName(String deptName) {
        this.deptName = deptName;
    }

    public List<Emp> getEmps() {
        return emps;
    }

    public void setEmps(List<Emp> emps) {
        this.emps = emps;
    }
}

//emp类
package cn.dawn.demo04.entity;

import cn.dawn.demo04.entity.Dept;

/**
 * Created by Dawn on 2018/2/26.
 */
public class Emp{
    private Integer empNo;
    private String empName;
    private Integer deptNo;
    private Dept dept;

    public Dept getDept() {
        return dept;
    }

    public void setDept(Dept dept) {
        this.dept = dept;
    }

    public Integer getEmpNo() {
        return empNo;
    }

    public void setEmpNo(Integer empNo) {
        this.empNo = empNo;
    }

    public String getEmpName() {
        return empName;
    }

    public void setEmpName(String empName) {
        this.empName = empName;
    }

    public Integer getDeptNo() {
        return deptNo;
    }

    public void setDeptNo(Integer deptNo) {
        this.deptNo = deptNo;
    }
}

 

  接口中使用的方法也就是个简单的根据部门编号返回该部门下全部员工的信息

  关注点不在这个方法,而在于一会怎么实现延迟加载和延迟加载的实际效果

 

    /*一对多多条sql*//*延迟加载*/
    public Dept findDeptnoALLEmpsMoreSql(Integer deptNo);

 

  接口同名的xml小配置中的代码我也放上去

 

    <!--一对多多条sql--><!--延迟加载--><!--缓存-->
    <resultMap id="DeptMoreSqlMapper" type="Dept">
        <id column="deptNo" property="deptNo"></id>
        <result column="deptName" property="deptName"></result>
        <collection property="emps" ofType="Emp" select="findDeptnoALLEmpsMoreSqlEmps" column="deptNo">
            <id column="empNo" property="empNo"></id>
            <result column="empName" property="empName"></result>
        </collection>
    </resultMap>
    <!--一对多多条sql--><!--延迟加载-->
    <select id="findDeptnoALLEmpsMoreSql" resultMap="DeptMoreSqlMapper">
        SELECT deptNo,deptName FROM dept WHERE deptNo=#{deptNo}
    </select>
    <select id="findDeptnoALLEmpsMoreSqlEmps" resultType="Emp">
        SELECT * FROM emp where deptNo=#{deptNo}
    </select>

 

  这个和之前用到的并没有什么不同,延迟加载在这儿并没有配置

  接下来是大配置的配置,

 

    <!--直接加载-->
    <!--不用配置就是直接加载-->
    <!--侵入式延迟-->
    <!--<settings>
        <setting name="lazyLoadingEnabled" value="true"/>
        <setting name="aggressiveLazyLoading" value="true"/>
    </settings>-->
    <!--深度延迟-->
    <!--<settings>
        <setting name="lazyLoadingEnabled" value="true"/>
        <setting name="aggressiveLazyLoading" value="false"/>
    </settings>-->

 

  我说一下这个怎么用,如果是你想要的方式,把他中文注释下的代码放开即可,

  lazyLoadingEnabled是延迟加载的开关,默认值是false,不开启,也就是直接加载————————》不要提有些mybatis的api,他里面写错了,你可以去试试

  aggressiveLazyLoading是侵入式的开关,默认值是true,开启,如果设置为false,则是深度加载的配置方法,侵入和深度使用有个前提,就是延迟加载的开关要开启,即lazyLoadingEnabled为true

 

=========================高贵的分割线=====================================================================================================================

  测试方法,我会把对应的延迟加载的配置放开,并附上运行结果,然后解释它

  直接加载略,简单讲述一下,就是大配置中的所有setting中的延迟配置都不用配,他只要调用该dao层的方法,就发送俩条sql,把所有查出来

  

  侵入式延迟加载,首先把对应的配置打开(即上面我说的那处,你把我写的那个注释取消掉)

  测试方法

    /*一延迟加载*/
    @Test
    public void t1LongtimeLoad(){
        SqlSession session= MyBatisUtils.getSession();

        IDeptDAO mapper = session.getMapper(IDeptDAO.class);
        Dept depts = mapper.findDeptnoALLEmpsMoreSql(1);

        System.out.println("====================================分割线===============================");
        /*侵入式延迟*/
        System.out.println(depts.getDeptName()+"================"+depts.getDeptNo());
        /*深度延迟*/
        /*System.out.println(depts.getEmps().size());*/


        session.close();

    }

  运行结果

    

  用了logback日志监控sql语句,发现在分割线下还有sql在执行了一遍,这就是我上面说的,侵入式延迟加载,在访问这个对象的任何属性时,才会走第二条sql,加载出这个对象里面关联的实体类的

  集合对象

 

 

  下面是深度延迟-------》同样把大配置的深度延迟打开,把上面那个注释掉

  测试方法

    /*一延迟加载*/
    @Test
    public void t1LongtimeLoad(){
        SqlSession session= MyBatisUtils.getSession();

        IDeptDAO mapper = session.getMapper(IDeptDAO.class);
        Dept depts = mapper.findDeptnoALLEmpsMoreSql(1);

        System.out.println("====================================分割线111111===============================");
        

        /*深度延迟*/
        System.out.println(depts.getDeptName()+"================"+depts.getDeptNo());
        System.out.println("====================================分割线222222===============================");
        System.out.println(depts.getEmps().size());

        /*侵入式延迟*/
        /*System.out.println(depts.getDeptName()+"================"+depts.getDeptNo());*/
        session.close();

    }

 

  运行结果

    

  这结果印证了我上面说的一句话,深度延迟,不访问这个对象的关联的那个对象集合时候,不做关联查询操作,不发第二条sql,一旦用到关联的那个对象集合的时候,他就做了 加载操作(关联查询),即发第二条SQL

 

 

 

写在最后的话,使用延迟加载,只有多条sql的关联查询才可以可以,单条的不可以,因为单条做的是表连接,直接一张表一下子就出来数据了

 

目录
相关文章
|
7月前
ssm(Spring+Spring mvc+mybatis)——updateDept.jsp
ssm(Spring+Spring mvc+mybatis)——updateDept.jsp
|
7月前
ssm(Spring+Spring mvc+mybatis)——showDept.jsp
ssm(Spring+Spring mvc+mybatis)——showDept.jsp
|
7月前
|
Java 关系型数据库 MySQL
基于ssm实现图书商城(spring+springmvc+mybatis)
基于ssm实现图书商城(spring+springmvc+mybatis)
|
7月前
|
Java 数据库连接 Maven
SSM框架整合:掌握Spring+Spring MVC+MyBatis的完美结合!
SSM框架整合:掌握Spring+Spring MVC+MyBatis的完美结合!
|
7月前
|
Java 数据库 数据安全/隐私保护
基于SSM框架实现管科类考研自我管理系统(分前后台spring+springmvc+mybatis+maven+jsp+jquery)
基于SSM框架实现管科类考研自我管理系统(分前后台spring+springmvc+mybatis+maven+jsp+jquery)
|
7月前
|
Java 数据库连接 数据库
【Mybatis】mybatis 是否支持延迟加载?
【Mybatis】mybatis 是否支持延迟加载?
|
6月前
|
缓存 NoSQL Java
在 SSM 架构(Spring + SpringMVC + MyBatis)中,可以通过 Spring 的注解式缓存来实现 Redis 缓存功能
【6月更文挑战第18天】在SSM(Spring+SpringMVC+MyBatis)中集成Redis缓存,涉及以下步骤:添加Spring Boot的`spring-boot-starter-data-redis`依赖;配置Redis连接池(如JedisPoolConfig)和连接工厂;在Service层使用`@Cacheable`注解标记缓存方法,指定缓存名和键生成策略;最后,在主配置类启用缓存注解。通过这些步骤,可以利用Spring的注解实现Redis缓存。
83 2
|
5月前
|
XML SQL Java
MyBatis 的延迟加载是如何实现的
MyBatis的延迟加载(懒加载)特性提高了性能,只在需要时加载关联数据。配置延迟加载需在`mybatis-config.xml`中设置`lazyLoadingEnabled`为`true`,`aggressiveLazyLoading`为`false`。实现原理基于代理对象,MyBatis为延迟加载属性创建代理,在访问时触发实际查询。代理通过Java动态代理实现,拦截方法调用,按需加载数据。
167 0
|
6月前
|
Java 数据库连接 Android开发
SSM框架——使用MyBatis Generator自动创建代码
SSM框架——使用MyBatis Generator自动创建代码
67 2
|
7月前
|
Java 关系型数据库 MySQL
SpringBoot整合JUnit、MyBatis、SSM
SpringBoot整合JUnit、MyBatis、SSM
49 4