mybatis延迟加载(懒加载)以及cglib动态代理

本文介绍了MyBatis中的延迟加载(懒加载)机制,旨在提高查询效率,避免不必要的资源浪费。通过实例展示了在User与Article两个实体类的关系中,如何设置延迟加载,使得在查询User时,仅在需要Article时才执行额外的数据库查询。同时,文章还探讨了MyBatis配置全局懒加载后的效果,并解释了采用懒加载后,MyBatis如何利用CGlib动态代理生成子类来实现延迟加载的功能。

有这么两个实体类

@Data
public class User implements Serializable {
    private Integer uid;
    private String name;
    private Integer age;
    private List<Article> articles;//测试collection
}
@Data
public class Article implements Serializable {
    private Integer aid;
    private Integer uid;
    private String title;
    private String content;
}

一个user可能对应多个article,查询user时有时并不需要把article也查出来(浪费资源),这样就要为两种情况分别写两个sql语句,一个只查user表,一个使用user join article联合两个表一次性把数据取出。

mybatis还支持延迟加载,在查询user的时候,只有用到article时才将article从数据库中取出

ArticleDao.xml

<mapper namespace="com.ssm.dao.ArticleDao">
    <sql id="articleColumns">aid, uid, title, content</sql>
    <select id="getArticleListByUserId" resultType="article"><!--别名大小写不敏感-->
        SELECT <include refid="articleColumns"/>
        FROM article
        where uid = #{uid}
    </select>
</mapper>

UserDao.xml

<mapper namespace="com.ssm.dao.UserDao">
    <!--使用嵌套查询-->
    <resultMap id="userMapUsingSelect" type="User">
        <id property="uid" column="uid" javaType="int" jdbcType="INTEGER"/>
        <result property="name" column="name" javaType="string" jdbcType="VARCHAR"/>
        <result property="age" column="age" javaType="int" jdbcType="INTEGER"/>
        <collection property="articles" column="uid" select="com.ssm.dao.ArticleDao.getArticleListByUserId"/>
    </resultMap>
    <select id="getUserByIdUsingSelect" resultMap="userMapUsingSelect">
        select uid, name, age
        from user
        where uid = #{uid}
    </select>
</mapper>

这里colloection要用select属性调用com.ssm.dao.ArticleDao命名空间下的getArticleListByUserId,分两次查询

ArticleDao.java

@Mapper
public interface ArticleDao {
     List<Article> getArticleListByUserId(@Param("uid") Integer userId);
}

UserDao.java

@Mapper
public interface UserDao {
    User getUserByIdUsingSelect(@Param("uid") Integer userId);
}

测试类

package com.ssm.dao;

import com.ssm.model.User;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import java.util.List;
import java.util.concurrent.TimeUnit;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration({"classpath:applicationcontext.xml"})
public class UserDaoTest {

    @Autowired private UserDao userMapper;

    @Test
    public void getUserByIdUsingSelect(){
        System.out.println(userMapper.getClass());
        System.out.println(userMapper.getClass().getSuperclass());
        User user = userMapper.getUserByIdUsingSelect(1);
        //这里千万别写System.out.println(user),因为会调用到article的toString方法
        System.out.println(user.getClass());//查看user类
        System.out.println(user.getClass().getSuperclass());//查看user父类

        //测试懒加载
        try {
            TimeUnit.SECONDS.sleep(10);
            System.out.println(user.getArticles());
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

没有设置懒加载时,mybatis取出user后立即又去查询article,控制台输出如下

可以看出mybatis查询出user后立刻去查询了article

在mybatis配置文件配置了全局懒加载后

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<!--    mybatis的全局配置文件中的标签是有先后顺序的,-->
<!--    按序分别是:properties、settings、typeAlliases、typeHandlers、objectFactory、plugins、environments、mappers。-->
    <settings>
        <!--开启全局的懒加载-->
        <setting name="lazyLoadingEnabled" value="true"/>
        <!--关闭立即加载,其实不用配置,默认为false-->
        <setting name="aggressiveLazyLoading" value="false"/>
    </settings>
</configuration>

再次运行测试类,控制台输出如下

可以看出十秒之后,当第一次调用user.getArticles时,mybatis才去查询article

还可以看出,没有采用懒加载时,mybatis将查询user表的结果封装成我们自己定义的实体类,即com.ssm.model.User。

采用懒加载后,mybatis使用了cglib动态代理,将查询user表的结果封装成一个动态生成的类(这里是com.ssm.model.User_$$_jvst9c1_0),这个类继承了com.ssm.model.User

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值