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

1、什么是延迟加载

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

    需求:

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

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

2、使用association实现延迟加载

    2.1 需求

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

    2.2 mapper.xml

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

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

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

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

    • 关联查询用户信息

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

    使用UserMapper.xml中的findUserById

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

    上边先去执行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

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

    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中配置:

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

    2.5.3 测试代码

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

    2.5.4 测试结果

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

3、延迟加载思考

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

    实现方法如下:

    定义两个mapper方法:

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

    实现思路:

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

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

    总之:

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

上一篇:团队作业(NABC的分析)


下一篇:美团HD(3)-加载分类导航数据