Mybatis的缓存机制

Mybatis的缓存机制

一、查询缓存

1.1、 什么是查询缓存

mybatis提供查询缓存,用于减轻数据压力,提高数据库性能。

mybaits提供一级缓存,和二级缓存。

一级缓存是基于SqlSession的作用范围,而二级缓存是基于mapper的namespace作用范围的
在这里插入图片描述

1.2、一级缓存

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

一级缓存的作用域是同一个SqlSession,在同一个sqlSession中两次执行相同的sql语句,第一次执行完毕会将数据库中查询的数据写到缓存(内存),第二次会从缓存中获取数据将不再从数据库查询,从而提高查询效率。当一个sqlSession结束后该sqlSession中的一级缓存也就不存在了。Mybatis默认开启一级缓存

一级缓存只是相对于同一个SqlSession而言。所以在参数和SQL完全一样的情况下,我们使用同一个SqlSession对象调用一个Mapper方法,往往只执行一次SQL,因为使用SelSession第一次查询后,MyBatis会将其放在缓存中,以后再查询的时候,如果没有声明需要刷新,并且缓存没有超时的情况下,SqlSession都会取出当前缓存的数据,而不会再次发送SQL到数据库。

1.3、一级缓存的生命周期有多长?

a、MyBatis在开启一个数据库会话时,会 创建一个新的SqlSession对象,SqlSession对象中会有一个新的Executor对象。Executor对象中持有一个新的PerpetualCache对象;当会话结束时,SqlSession对象及其内部的Executor对象还有PerpetualCache对象也一并释放掉。

b、如果SqlSession调用了close()方法,会释放掉一级缓存PerpetualCache对象,一级缓存将不可用。

c、如果SqlSession调用了clearCache(),会清空PerpetualCache对象中的数据,但是该对象仍可使用。

d、SqlSession中执行了任何一个update操作(update()、delete()、insert()) ,都会清空PerpetualCache对象的数据,但是该对象可以继续使用

1.4、怎么判断某两次查询是完全相同的查询?

mybatis认为,对于两次查询,如果以下条件都完全一样,那么就认为它们是完全相同的两次查询。

2.1 传入的statementId

2.2 查询时要求的结果集中的结果范围

2.3. 这次查询所产生的最终要传递给JDBC java.sql.Preparedstatement的Sql语句字符串(boundSql.getSql() )

2.4 传递给java.sql.Statement要设置的参数值

1.5、二级缓存

MyBatis的二级缓存是Application级别的缓存,它可以提高对数据库查询的效率,以提高应用的性能。

MyBatis的缓存机制整体设计以及二级缓存的工作模式
在这里插入图片描述
二级缓存是mapper级别的缓存,多个SqlSession去操作同一个Mapper的sql语句,多个SqlSession去操作数据库得到数据会存在二级缓存区域,多个SqlSession可以共用二级缓存,二级缓存是跨SqlSession的。

二级缓存是多个SqlSession共享的,其作用域是mapper的同一个namespace,不同的sqlSession两次执行相同namespace下的sql语句且向sql中传递参数也相同即最终执行相同的sql语句,第一次执行完毕会将数据库中查询的数据写到缓存(内存),第二次会从缓存中获取数据将不再从数据库查询,从而提高查询效率。Mybatis默认没有开启二级缓存需要在setting全局参数中配置开启二级缓存。
sqlSessionFactory层面上的二级缓存默认是不开启的,二级缓存的开启需要进行配置,实现二级缓存的时候,MyBatis要求返回的POJO必须是可序列化的。 也就是要求实现Serializable接口,配置方法很简单,只需要在映射XML文件配置就可以开启缓存了,如果我们配置了二级缓存就意味着:

  • 映射语句文件中的所有select语句将会被缓存。
  • 映射语句文件中的所欲insert、update和delete语句会刷新缓存。
  • 缓存会使用默认的Least Recently Used(LRU,最近最少使用的)算法来收回。
  • 根据时间表,比如No Flush Interval,(CNFI没有刷新间隔),缓存不会以任何时间顺序来刷新。
  • 缓存会存储列表集合或对象(无论查询方法返回什么)的1024个引用
  • 缓存会被视为是read/write(可读/可写)的缓存,意味着对象检索不是共享的,而且可以安全的被调用者修改,不干扰其他调用者或线程所做的潜在修改。
  • 如果缓存中有数据就不用从数据库中获取,大大提高系统性能。

二、一级缓存

2.1 一级缓存工作原理

下图是根据id查询用户的一级缓存图解
在这里插入图片描述
第一次发起查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,如果没有,从数据库查询用户信息。
得到用户信息,将用户信息存储到一级缓存中。

如果sqlSession去执行commit操作(执行插入、更新、删除),清空SqlSession中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读。
第二次发起查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,缓存中有,直接从缓存中获取用户信息。

2.2 一级缓存测试

mybatis默认支持一级缓存,不需要在配置文件去配置。

按照上示原理步骤去实现

public class TestService {
    private InputStream in = null;
    private SqlSession sqlSession = null;
    private UserInfoMapper userInfoMapper;
    @Before
    public void init() throws IOException {
        //读取配置文件
        in = Resources.getResourceAsStream("MybatisConfig.xml");
        //创建工厂构建者
        SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
        //构建工厂
        SqlSessionFactory factory = builder.build(in);
        //根据工厂获取session
        sqlSession = factory.openSession();
        //sqlSession构建代理对象
        userInfoMapper = sqlSession.getMapper(UserInfoMapper.class);
        System.out.println();
    }

    @After
    public void after(){
        if(sqlSession != null){
            sqlSession.close();
        }
        if(in!= null){
            try {
                in.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    @Test
    public void firstCacheDemo(){
        UserInfo user1 = userInfoMapper.findUserById(1);
        UserInfo user2 = userInfoMapper.findUserById(1);
        System.out.println("测试一级缓存===");
        System.out.println(user1 == user2);

        UserInfo updateUser = userInfoMapper.findUserById(2);
        updateUser.setStaffName(updateUser.getStaffName()+"更新");
        userInfoMapper.updateUser(updateUser);
        sqlSession.commit();

        //这个方法也可以删除一级缓存
        sqlSession.clearCache();

        userInfoMapper.updateUser(user1);

        UserInfo user3 = userInfoMapper.findUserById(1);

        System.out.println("执行commit后比较缓存===");
        System.out.println(user2 == user3);
    }
}

执行结果
在这里插入图片描述

三、二级缓存

3.1 实现原理

在这里插入图片描述

首先开启mybatis的二级缓存。

sqlSession1去查询用户id为1的用户信息,查询到用户信息会将查询数据存储到二级缓存中。

如果SqlSession3去执行相同 mapper下sql,执行commit提交,清空该 mapper下的二级缓存区域的数据

sqlSession2去查询用户id为1的用户信息,去缓存中找是否存在数据,如果存在直接从缓存中取出数据。

二级缓存与一级缓存区别,二级缓存的范围更大,多个sqlSession可以共享一个UserMapper的二级缓存区域。数据类型仍然为HashMap

UserMapper有一个二级缓存区域(按namespace分,如果namespace相同则使用同一个相同的二级缓存区),其它mapper也有自己的二级缓存区域(按namespace分)。

每一个namespace的mapper都有一个二缓存区域,两个mapper的namespace如果相同,这两个mapper执行sql查询到数据将存在相同的二级缓存区域中。

3.2 开启二级缓存

mybaits的二级缓存是mapper范围级别,除了在SqlMapConfig.xml设置二级缓存的总开关,还要在具体的mapper.xml中开启二级缓存。

在核心配置文件SqlMapConfig.xml中加入

<setting name="cacheEnabled"value="true"/>
	<!-- 全局配置参数,需要时再设置 -->
    <settings>
	<!-- 开启二级缓存  默认值为true -->
 	<setting name="cacheEnabled" value="true"/>
</settings> 
属性名描述允许值默认值
cacheEnabled对在此配置文件下的所有cache 进行全局性开/关设置。true/falsetrue

在UserMapper.xml中开启二缓存,UserMapper.xml下的sql执行完成会存储到它的缓存区域(HashMap)。

<mapper namespace="cn.hpu.mybatis.mapper.UserMapper">
<!-- 开启本mapper namespace下的二级缓存 -->
<cache></cache>

调用pojo类实现序列化接口

@Data
public class UserInfo implements Serializable {

    private Integer id;

    private String staffId;

    private String staffName;

    /**
     * 逻辑删除
     */
    private boolean enabled;
}

3.3 二级缓存测试

二级缓存需要查询结果映射的pojo对象实现java.io.Serializable接口实现序列化和反序列化操作,注意如果存在父类、成员pojo都需要实现序列化接口。
pojo类实现序列化接口是为了将缓存数据取出执行反序列化操作,因为二级缓存数据存储介质多种多样,不一定在内存有可能是硬盘或者远程服务器。

// 二级缓存测试
 
   @Test
 
   public void testCache2() throws Exception {
 
      SqlSessionsqlSession1 = sqlSessionFactory.openSession();
 
      SqlSessionsqlSession2 = sqlSessionFactory.openSession();
 
      SqlSessionsqlSession3 = sqlSessionFactory.openSession();
 
      // 创建代理对象
 
      UserMapperuserMapper1 = sqlSession1.getMapper(UserMapper.class);
 
      // 第一次发起请求,查询id为1的用户
 
      Useruser1 = userMapper1.findUserById(1);
 
      System.out.println(user1);
 
     
 
      //这里执行关闭操作,将sqlsession中的数据写到二级缓存区域
 
      sqlSession1.close();
 
     
 
      //使用sqlSession3执行commit()操作
 
      UserMapperuserMapper3 = sqlSession3.getMapper(UserMapper.class);
 
      Useruser  = userMapper3.findUserById(1);
 
      user.setUsername("张明明");
 
      userMapper3.updateUser(user);
 
      //执行提交,清空UserMapper下边的二级缓存
 
      sqlSession3.commit();
 
      sqlSession3.close();
 
     
 
      UserMapperuserMapper2 = sqlSession2.getMapper(UserMapper.class);
 
      // 第二次发起请求,查询id为1的用户
 
      Useruser2 = userMapper2.findUserById(1);
 
      System.out.println(user2);
 
 
 
      sqlSession2.close();
 
   }

3.4 useCache配置禁用二级缓存

在statement中设置useCache=false可以禁用当前select语句的二级缓存,即每次查询都会发出sql去查询,默认情况是true,即该sql使用二级缓存。

<selectid="findOrderListResultMap" resultMap="ordersUserMap" useCache="false">

总结:针对每次查询都需要最新的数据sql,要设置成useCache=false,禁用二级缓存。

3.5 mybatis刷新缓存(就是清空缓存)

在mapper的同一个namespace中,如果有其它insert、update、delete操作数据后需要刷新缓存,如果不执行刷新缓存会出现脏读。

设置statement配置中的flushCache=“true” 属性,默认情况下为true即刷新缓存,如果改成false则不会刷新。使用缓存时如果手动修改数据库表中的查询数据会出现脏读。

如下:

<insertid="insertUser" parameterType="cn.itcast.mybatis.po.User" flushCache="true">

总结:一般下执行完commit操作都需要刷新缓存,flushCache=true表示刷新缓存默认情况下为true,我们不用去设置它,这样可以避免数据库脏读。

3.6 Mybatis Cache参数

flushInterval(刷新间隔)可以被设置为任意的正整数,而且它们代表一个合理的毫秒形式的时间段。默认情况是不设置,也就是没有刷新间隔,缓存仅仅调用语句时刷新。

size(引用数目)可以被设置为任意正整数,要记住你缓存的对象数目和你运行环境的可用内存资源数目。默认值是1024。

readOnly(只读)属性可以被设置为true或false。只读的缓存会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。这提供了很重要的性能优势。可读写的缓存会返回缓存对象的拷贝(通过序列化)。这会慢一些,但是安全,因此默认是false。

如下例子:

<cache  eviction="FIFO" flushInterval="60000"  size="512" readOnly="true"/>

这个更高级的配置创建了一个 FIFO 缓存,并每隔 60 秒刷新,存数结果对象或列表的 512 个引用,而且返回的对象被认为是只读的,因此在不同线程中的调用者之间修改它们会导致冲突。可用的收回策略有, 默认的是 LRU:

  • LRU – 最近最少使用的:移除最长时间不被使用的对象。
  • FIFO – 先进先出:按对象进入缓存的顺序来移除它们。
  • SOFT – 软引用:移除基于垃圾回收器状态和软引用规则的对象。
  • WEAK – 弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值