Mybatis学习记录(七)----Mybatis延迟加载

1、什么是延迟加载

    resultMap可以实现高级映射(使用association、collection实现一对一及一对多映射),association、collection具备延迟加载功能。

    需求:

    如果查询订单并且关联查询用户信息。如果先查询订单信息即可满足要求,当我们需要查询用户信息时再查询用户信息。把对用户信息的按需去查询就是延迟加载。

    延迟加载:先从单表查询、需要时再从关联表去关联查询,大大提高 数据库性能,因为查询单表要比关联查询多张表速度要快。

2、使用association实现延迟加载

    2.1 需求

    查询订单并且关联查询用户信息

    2.2 mapper.xml

    需要定义两个mapper的方法对应的statement。

    • 只查询订单信息
    SELECT * FROM orders

    在查询订单的statement中使用association去延迟加载(执行)下边的satatement(关联查询用户信息)

    clip_image001

    • 关联查询用户信息

    通过上边查询到的订单信息中user_id去关联查询用户信息

    使用UserMapper.xml中的findUserById

    clip_image002

    上边先去执行findOrdersUserLazyLoading,当需要去查询用户的时候再去执行findUserById,通过resultMap的定义将延迟加载执行配置起来。

    2.3 延迟加载resultMap

    使用association中的select指定延迟加载去执行的statement的id。

    <!-- 延迟加载的resultMap -->
        <resultMap type="com.joe.mybatis.entities.Orders" id="OrdersUserLazyLoadingResultMap" >
        
            <!-- 配置映射的订单信息 -->
            <!-- id:指定查询列中的唯一标示,在这里是指订单中的唯一标示,如果有多列组成唯一标示,就需要配置多个id -->
            <!-- column:订单信息表的唯一标识列 property:订单信息的唯一标识列所映射到Orders类中的属性名 -->
            <id column="id" property="id" />
    
            <result column="user_id" property="userId" />
            <result column="number" property="number" />
            <result column="createtime" property="createtime" />
            <result column="note" property="note" />
    
            <!-- 配置映射关联的用户信息 -->
            <!-- association:用户映射关联查询单个对象的信息 property:要将关联查询对象映射到Orders类中的属性名 javaType:property属性所指定的列的类型 -->
            <!-- 
                select:指定延迟加载需要执行的statement的id(是根据user_id查询用户信息的statement);
                使用UserMapper.xml文件中的findUserById(传入user_id)进行查询,如果findUserById不在本Mapper中,需要加上namespace
                column:订单信息中关联用户信息的列,是user_id
                关联查询的sql理解为:
                SELECT
                    orders.*, (
                        SELECT
                            username
                        FROM
                            `user`
                        WHERE
                            `user`.id = orders.user_id
                    ) username,
                    (
                        SELECT
                            sex
                        FROM
                            `user`
                        WHERE
                            `user`.id = orders.user_id
                    ) sex
                FROM
                    orders
             -->
            <association property="user" javaType="com.joe.mybatis.entities.User" 
                        select="com.joe.mybatis.mapper.UserMapper.findUserById" column="user_id">
    
                
    
    
            </association>
        
        </resultMap>

    2.4 mapper.java

    clip_image003

    2.5 测试

    2.5.1 测试思路:

    • 执行上边mapper方法(findOrdersUserLazyLoading),内部去调用cn.itcast.mybatis.mapper.OrdersMapperCustom中的findOrdersUserLazyLoading只查询orders信息(单表)。
    • 在程序中去遍历上一步骤查询出的List<Orders>,当我们调用Orders中的getUser方法时,开始进行延迟加载。
    • 延迟加载,去调用UserMapper.xml中findUserbyId这个方法获取用户信息。

    2.5.2 延迟加载配置

    mybatis默认没有开启延迟加载,需要在SqlMapConfig.xml中setting配置。

    在mybatis核心配置文件中配置:

    lazyLoadingEnabled、aggressiveLazyLoading

    设置项 描述 允许值 默认值
    lazyLoadingEnabled

    全局性设置懒加载。如果设为‘false’,则所有相关联的都会被初始化加载。

    true | false

    false

    aggressiveLazyLoading

    当设置为‘true’的时候,懒加载的对象可能被任何懒属性全部加载。否则,每个属性都按需加载。

    true | false

    true

     

    在SqlMapConfig.xml中配置:

    clip_image004

    2.5.3 测试代码

    clip_image005

    2.5.4 测试结果

    clip_image006

3、延迟加载思考

    不使用mybatis提供的association及collection中的延迟加载功能,如何实现延迟加载??

    实现方法如下:

    定义两个mapper方法:

    • 查询订单列表
    • 根据用户id查询用户信息

    实现思路:

    先去查询第一个mapper方法,获取订单信息列表

    在程序中(service),按需去调用第二个mapper方法去查询用户信息。

    总之:

    使用延迟加载方法,先去查询简单的sql(最好单表,也可以关联查询),再去按需要加载关联查询的其它信息。

原文地址:https://www.cnblogs.com/doctorJoe/p/5291208.html