3-Mybatis-延迟加载与缓存

延迟加载

1. 需求:

查询订单并关联查询用户信息
- 先对orders表进行单表查询,然后通过外键user_id对user表进行查询
- 定义两个mapper方法对应的statement

1.1只查询订单信息

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

1.2关联查询用户信息

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

2. SQL

SELECT orders.*,
(SELECT `user`.user_name
    FROM `user` 
    WHERE `user`.user_id = orders.user_id)user_name,
(SELECT `user`.user_birth
    FROM `user` 
    WHERE `user`.user_id = orders.user_id)user_birth
FROM orders

3. mapper.xml

<!-- 延迟加载:查询订单并关联查询用户信息 -->
<resultMap type="com.wzw.mybatis.model.Orders" id="OrderAndUserResultMap">
    <!-- 配置订单信息 -->
    <id column="orders_id" property="orders_id"/>
    <result column="number" property="number"/>
    <!-- <result column="user_id" property="user_id"/> -->
    <!-- 配置用户信息,实现延迟加载 
        select属性:指定延迟加载需要执行的statement的id
        column属性:订单信息关联查询用户信息查询的列,user_id
    -->
    <association property="user" javaType="com.wzw.mybatis.model.User" 
    select="com.wzw.mybatis.mapper.UserMapper.selectUserById" column="user_id">
    </association>
</resultMap>
<select id="selectOrdersAndUserLazyLoading" resultMap="OrderAndUserResultMap">
    SELECT * FROM orders
</select>

UserMapper.xml
<mapper namespace="com.wzw.mybatis.mapper.UserMapper">
<!-- 根据user_id查询用户信息 -->
    <select id="selectUserById" parameterType="java.lang.Integer" resultType="user">
        SELECT * FROM `user` WHERE user_id = #{id}  
    </select>
</mapper>

4. mapper.java

//延迟加载:查询订单并关联查询用户信息
public List<Orders> selectOrdersAndUserLazyLoading() throws IOException;

5. SQLMapConfig.xml延迟加载配置

<!-- 延迟加载 -->
<settings>
    <setting name="lazyLoadingEnabled" value="true"/><!-- 默认false -->
    <setting name="aggressiveLazyLoading" value="false"/><!-- 默认true -->
</settings>

6. 测试代码

OrdersMapper mapper = sqlSession.getMapper(OrdersMapper.class);
//查询订单信息
List<Orders> ordeList = mapper.selectOrdersAndUserLazyLoading();
//对订单集合遍历
for (Orders orders : ordeList) {
    //按需加载,查询用户信息
    User user = orders.getUser();
    System.out.print(user);
}

缓存

这里写图片描述
一级缓存是SqlSession级别的缓存。在操作数据库时需要创建SqlSession对象。在SqlSession对象中有一个数据结构(HashMap)用于存储缓存数据。不同的SqlSession之间的缓存数据区域(HashMap)互不影响。

二级缓存是mapper级别的缓存。多个SqlSession执行同一个mapper的SQL语句,多个SqlSession共用二级缓存,二级缓存是跨SqlSession的。

1. 一级缓存

mybatis默认开启一级缓存

  • 第一次对user_id=1的用户进行查询,首先看一级缓存中是否存在,不存在,此时将查询信息user存入SqlSession的一级缓存中
  • 此时对user执行更新,删除等需要SqlSession.commit()操作时,清空SqlSession中一级缓存内的数据
  • 第三次对user_id=1的用户进行查询,首先看一级缓存中是否存在,不存在,此时将查询信息user存入SqlSession的一级缓存中
  • 第四次对user_id=1的用户进行查询,发现一级缓存中存在数据,从一级缓存中读取数据

2. 二级缓存

按mapper区分,每一个mapper都有自己的二级缓存区域,此处mapper按namespace区分。既每一个namespace的mapper有一个二级缓存区域。如果两个mapper的namespace相同,则执行这两个mapper中的SQL语句查询到的数据都存储在同一个二级缓存中。

2.1 开启二级缓存

SQLMapConfig.xml

<settings>
    <!-- 延迟加载 -->
    <setting name="lazyLoadingEnabled" value="true"/><!-- 默认false -->
    <setting name="aggressiveLazyLoading" value="false"/><!-- 默认true -->
    <!-- 配置二级缓存,默认开启 -->
    <setting name="cacheEnabled" value="true"/>
</settings>

UserMapper.xml

<!-- 开启二级缓存 -->
<cache></cache>

2.2 pojo类实现序列号Serializable

public class User implements Serializable

2.3 禁用二级缓存

useCache=false,在statement中配置该条命令,可禁用二级缓存,默认true。
针对每次查询都需要最新的数据

<!-- 根据user_id查询用户信息 -->
<select id="selectUserById" parameterType="java.lang.Integer" resultType="user" useCache="false">
    SELECT * FROM `user` WHERE user_id = #{id}  
</select>

2.4 刷新缓存(清空缓存)

flushCache=true,默认为true。在insert、update、delete时,每次操作后需要刷新。避免出现脏读。

2.5 Cache标签参数

  • flushInterval(刷新间隔):可设置为任意的正整数,表示一个毫秒为单位的时间段。默认不设置(无刷新间隔),缓存在调用语句时刷新。
  • size(引用数目):可以设置为任意整数,要记住你缓存的对象数目和你运行环境的可用内存资源数目。默认为1024。
  • readOnly(只读):true/false。只读的缓存会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。可读写的缓存会返回对象的拷贝(通过序列号)。这会慢一些,但是安全,因此默认false。

3 mybatis整合ehcache

Mybatis提供了一个cache接口,如果要实现自己的缓存逻辑,实现cache接口开发即可。
Mybatis和ehcache整合,mybatis和ehcache整合包中提供了一个cache接口的实现类。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值