1. 什么是延迟加载

举个例子:

  如果查询订单并且关联查询用户信息。如果先查询订单信息即可满足要求,当我们需要查询用户信息时再查询用户信息。把对用户信息的按需去查询就是延迟加载。 所以延迟加载即先从单表查询、需要时再从关联表去关联查询,大大提高数据库性能,因为查询单表要比关联查询多张表速度要快。

我们来对比一下:

关联查询:

SELECT orders.*, user.username FROM orders, USER WHERE orders.user_id = user.id 

延迟加载相当于:

SELECT orders.*,
(SELECT username FROM USER WHERE orders.user_id = user.id)username FROM orders

  所以这就比较直观了,也就是说,我把关联查询分两次来做,而不是一次性查出所有的。第一步只查询单表orders,必然会查出orders中的一个user_id字段,然后我再根据这个user_id查user表,也是单表查询。下面来总结一下如何使用这个延迟加载

2. 使用association实现延迟加载

  resultMap可以实现高级映射(使用association、collection实现一对一及一对多映射),其实association和collection还具备延迟加载的功能,这里我就拿association来说明,collection和association使用的方法都是一样的。需求就是上面提到的,查询订单并且关联查询用户,查询用户使用延迟加载。 
  由上面的分析可知,延迟加载要查询两次,第二次是按需查询,之前一对一关联查询的时候只需要查一次,把订单和用户信息都查出来了,所以只要写一个mapper即可,但是延迟加载查两次,所以理所当然要有两个mapper。

2.1 两个mapper.xml

  需要定义两个mapper的方法对应的statement。先来分析一下思路:

只查询订单信息的statement,使用resultMap通过查询到的订单信息中的user_id去查询用户信息的statement,得到用户定义的resultMap将两者关联起来,即用订单信息user_id去查用户

下面来实现这个思路: 
1. 在OrdersMapper.xml文件中配置只查询订单信息的statement:

<select id="findOrdersUserLazyLoading" resultMap="OrdersUserLazyLoadingResultMap">

SELECT * FROM orders

</select>

2. 在UserMapper.xml文件中配置只查询用户信息的statement:

<select id="findUserById" parameterType="int" resultType="user">

    select * from user where id = #{id}

</select>

3. 定义上面那个resultMap:

2.2 延迟加载的配置

mybatis默认没有开启延迟加载,需要在SqlMapConfig.xml中setting配置。在SqlMapConfig.xml中的一些配置,有一个<settings>,当时没说,这里就派上用场了,可以通过这个标签来配置一下延迟加载。

 <settings>

     <!-- 打开延迟加载的开关 -->

     <setting name="lazyLoadingEnabled" value="true"/>

     <!-- 将积极加载改为消极加载,即延迟加载 -->

     <setting name="aggressiveLazyLoading" value="false"/>

 </settings>

2.3 mapper.java

  OrdersMapper中添加mapper接口:

 public interface UserMapperOrders {

     //查询订单,关联用户查询,用户查询用的是延迟加载

     public List<Orders> findOrdersUserByLazy() throws Exception;

 }

2.4 代码测试

 @Test

    public void testFindOrdersUserByLazy() {

       SqlSession session = sqlSessionFactory.openSession();

       OrdersMapper ordersMapper = session.getMapper(OrdersMapper.class);

       List<Orders> list = ordersMapper.findOrdersUserByLazy();

       for (Orders orders : list) {

           System.out.println(orders);

           User user = orders.getUser();

           System.out.println(user);

       }

    }

看一下执行结果:

  执行结果很明显,使用了延迟加载,将关联查询分成了两次单表查询,但是有个奇怪的地方,就是第二次查用户的时候,并没有发sql,而是直接就拿到了,其实这就是mybatis中的一级缓存

mybatis的延迟加载的更多相关文章

  1. mybatis的延迟加载、一级缓存、二级缓存

    mybatis的延迟加载.一级缓存.二级缓存 mybatis是什么? mybatis是一个持久层框架,是apache下的开源项目,前身是itbatis,是一个不完全的ORM框架,mybatis提供输入 ...

  2. Mybatis之延迟加载机制

    1.  延迟加载的含义: 用到的时候才会去进行相关操作 2.  延迟加载的例子: 2.1 spring的BeanFactory,在getBean()的时候才创建Bean 2.2 物理分页查询,只有点击 ...

  3. Mybatis的延迟加载和缓存

    1. MyBatis中的延迟加载,也称为懒加载,是指在进行关联查询时,按照设置延迟加载规则推迟对关联对象的select查询.延迟加载可以有效的减少数据库压力.       注意:MyBatis的延迟加 ...

  4. Mybatis 测试延迟加载

    在学习mybatis的延迟加载时,对 lazyLoadingEnabled 和 aggressiveLazyLoading 的区别并不理解,特别是对查询的条件不同时,执行的查询语句也不一样,所以还是测 ...

  5. Mybatis的延迟加载和立即加载

    Mybatis的延迟加载和立即加载 示例:在一对多中,当我们有一个用户,他有100个帐户 问题1:在查询用户时,要不要把关联的账户查出来? 问题2:在查询账户时,要不要把关联的用户信息查出来? 问题1 ...

  6. MyBatis学习--延迟加载

    简介 在resultMap可以实现高级映射(使用association.collection实现一对一及一对多映射),association.collection具备延迟加载功能.例如:我们查询订单并 ...

  7. 8.Mybatis的延迟加载

    在Mybatis中的延迟加载只有resultMap可以实现,ResultMap 可以实现高级映射(association,collection可以实现一对1和一对多的映射),他们具有延迟加载的功能,r ...

  8. 七 mybatis的延迟加载

    1       延迟加载 1.1     什么是延迟加载 resultMap可以实现高级映射(使用association.collection实现一对一及一对多映射),association.coll ...

  9. mybatis实现延迟加载多对一

    1.数据库表 CREATE TABLE `country` ( `cid` ) NOT NULL AUTO_INCREMENT COMMENT '国家id', `cname` ) COLLATE ut ...

随机推荐

  1. 【PHP】数组用法(转)

    摘要: 说明数组遍历方法foreach,while,for,推荐使用foreach(PHP内部实现,简单速度最快,还可以遍历类属性).以及一些常用方法current,prev,next,end,key ...

  2. ue4粒子实现流血效果

    ---恢复内容开始--- 动作/射击游戏中,击中角色时常常伴随着血花效果,增强打击感的同时,也方便了玩家对命中与否的判断. 血液效果分两块,首先是受伤部位在受击瞬间产生血雾粒子,然后在身体.地面.墙面 ...

  3. 1. 数字根(Digital Root)

    数字根(Digital Root)就是把一个自然数的各位数字相加,再将所得数的各位数字相加,直到所得数为一位数字为止.而这个一位数便是原来数字的数字根.例如: 198的数字根为9(1+9+8=18,1 ...

  4. git分支管理之创建与合并分支

    在版本回退里,你已经知道,每次提交,Git都把它们串成一条时间线,这条时间线就是一个分支.截止到目前,只有一条时间线,在Git里,这个分支叫主分支,即master分支.HEAD严格来说不是指向提交,而 ...

  5. html5 textarea 文本框根据输入内容自适应高度

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...

  6. jQuery相关知识总结一

    1day-jquery 1. 1 jQuery 1概念 * JavaScript(ECMA/DOM/BOM)在实际开发中,使用比较麻烦,有浏览器兼容问题. * JavaScript类库(JS库) 的目 ...

  7. java IO流整理

    Java流操作有关的类或接口: Java流类图结构: 流的概念和作用 流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象.即数据在两设备间的传输称为流,流的本质是数据传输,根据数据传输 ...

  8. Markdown公式编辑

    一.公式使用参考 1.如何插入公式 行中公式(放在文中与其它文字混编)可以用如下方法表示:$ 数学公式 $ 独立公式可以用如下方法表示:$$ 数学公式 $$ 自动编号的公式可以用如下方法表示: 若需要 ...

  9. InnoDB: ERROR: the age of the last checkpoint

    --InnoDB: ERROR: the age of the last checkpoint ---------------------------------------------------- ...

  10. ajax轮询实时获取数据

    最近做一个评论功能时,想要实现实时异步刷新评论功能,于是使用了ajax轮询,这里简单记录一下ajax轮询的原理及使用方法. ajax轮询的原理就是客户端定时向服务端发送ajax请求,服务器接到请求后马 ...