MyBatis|缓存机制--一级缓存[转]

在这里插入图片描述

1. MyBatis缓存

1.1 缓存介绍

就是将用户经常查询的数据的结果的一个保存,保存到一个内存中(缓存就是内存中的一个对象),用户在查询的时候就不用到数据库文件中查询(磁盘),从而减少与数据库的交付次数提高了响应速度,解决了并发系统的西能问题。

1.2 MyBatis缓存分类

MyBatis提供了一级缓存和二级缓存

一级缓存:也称为本地缓存,用于保存用户在一次会话过程中查询的结果,用户一次会话中只能使用一个sqlSession,一级缓存是自动开启的,不允许关闭。
二级缓存:也称为全局缓存,是mapper级别的缓存,是针对一个表的查结果的存储,可以共享给所有针对这张表的查询的用户。也就是说对于mapper级别的缓存不同的sqlsession是可以共享的。

1.3 什么是会话

会话就是一次完整的交流,再一次交流过程中包含多次请求响应,而发送的请求都是同一个用户,SqlSession就是用户与数据库进行一次会话过程中使用的接口。

2. MyBatis一级缓存

2.1 一级缓存介绍

在应用运行过程中,在一次数据库会话中,执行多次查询条件完全相同的SQL,会优先命中一级缓存,避免直接对数据库中直接查询。
在这里插入图片描述
每个SqlSession中都持有Excutor,每个Excutor中有一个LocalCache。当用户发起询问时,MyBatis根据当前执行的语句生成MappedStatement,在Local Cache进行查询,如果缓存命中的话,直接返回结果给用户,如果缓存没有命中的话,查询数据库,结果写入Local Cache,最后返回结果给用户。
在这里插入图片描述

2.2 源码分析

SqlSession: 对外提供了用户和数据库之间交互需要的所有方法,隐藏了底层的细节。默认实现类是DefaultSqlSession。
Executor: SqlSession向用户提供操作数据库的方法,但和数据库操作有关的职责都会委托给Executor。Executor有两个实现类,和一级缓存关联的是BaseExecutor。
BaseExecutor: BaseExecutor是一个实现了Executor接口的抽象类,定义若干抽象方法,在执行的时候,把具体的操作委托给子类进行执行。
PerpetualCache:对Cache接口最基本实现,内部持有HashMap,对一级缓存的操作实则是对HashMap的操作。

SqlSession初始化时会创建Executor的实例,Mybatis默认使用的是SimpleExecutor,初始化代码如下所示:

public Executor newExecutor(Transaction transaction, ExecutorType executorType) {
    executorType = executorType == null ? defaultExecutorType : executorType;
    executorType = executorType == null ? ExecutorType.SIMPLE : executorType;
    Executor executor;
    if (ExecutorType.BATCH == executorType) {
      executor = new BatchExecutor(this, transaction);
    } else if (ExecutorType.REUSE == executorType) {
      executor = new ReuseExecutor(this, transaction);
    } else {
      executor = new SimpleExecutor(this, transaction);
    }
   //如果启用二级缓存,使用CahingExecutor装饰类
    if (cacheEnabled) {
      executor = new CachingExecutor(executor);
    }
    executor = (Executor) interceptorChain.pluginAll(executor);
    return executor;
  }

SqlSession在提交的时候会清空本地缓存,因为commit操作一般对应插入、更新或者删除操作,清空缓存防止读取脏数据。

  @Override
  public void commit(boolean required) throws SQLException {
    if (closed) {
      throw new ExecutorException("Cannot commit, transaction is already closed");
    }
    clearLocalCache();
    flushStatements();
    if (required) {
      transaction.commit();
    }
  }

SqlSession的insert方法和delete方法,都会统一走update的流程。

@Override
public int insert(String statement, Object parameter) {
    return update(statement, parameter);
  }
   @Override
  public int delete(String statement) {
    return update(statement, null);
}

update方法也是委托给了Executor执行。BaseExecutor的执行方法如下所示。

@Override
public int update(MappedStatement ms, Object parameter) throws SQLException {
    ErrorContext.instance().resource(ms.getResource()).activity("executing an update").object(ms.getId());
    if (closed) {
      throw new ExecutorException("Executor was closed.");
    }
    clearLocalCache();
    return doUpdate(ms, parameter);
}

2.3 一级缓存失效的原因

  • 同一个用户使用不同的SqlSession对象导致无法看到一级缓存工作。
              DeptDao dao1 =   session.getMapper(DeptDao.class);
              Dept dept1 = dao1.findByDeptNo(10);
              System.out.println("第一次查询的部门对象地址 "+dept1);
              
              //新建一个SqlSession
              SqlSession session2 =  factory.openSession();
              DeptDao dao2 =   session2.getMapper(DeptDao.class);
              Dept dept2 = dao2.findByDeptNo(10);
              System.out.println("第二次查询的部门对象地址 "+dept2);
  • 在一个SqlSession中使用条件查询不同一级缓存也会失效。
            DeptDao dao1 =   session.getMapper(DeptDao.class);
              Dept dept1 = dao1.findByDeptNo(10);
              System.out.println("第一次查询的部门对象地址 "+dept1);
              
             
              DeptDao dao2 =   session.getMapper(DeptDao.class);
              Dept dept2 = dao2.findByDeptNo(20);
              System.out.println("第二次查询的部门对象地址 "+dept2);
  • 在一个SqlSession使用相同条件,但是,此时在查询之间进行数据修改操作会导致一级缓存失效。
DeptDao dao1 =   session.getMapper(DeptDao.class);
              Dept dept1 = dao1.findByDeptNo(10);
              System.out.println("第一次查询的部门对象地址 "+dept1);
              
              //执行数据库修改
              Dept dept = new Dept(40, "测试部门", "深圳");
              dao1.deptAdd(dept);
             
              DeptDao dao2 =   session.getMapper(DeptDao.class);
              Dept dept2 = dao2.findByDeptNo(10);
              System.out.println("第二次查询的部门对象地址 "+dept2);
              
              session.commit();
  • 在一个SqlSession使用相同查询条件此时手动刷新缓存时导致一级缓存失败。
    DeptDao dao1 =   session.getMapper(DeptDao.class);
              Dept dept1 = dao1.findByDeptNo(10);
              System.out.println("第一次查询的部门对象地址 "+dept1);
              
              //手动刷新用户一级缓存,导致用户一级缓存原有的内容消失掉
              session.clearCache();
              
              DeptDao dao2 =   session.getMapper(DeptDao.class);
              Dept dept2 = dao2.findByDeptNo(10);
              System.out.println("第二次查询的部门对象地址 "+dept2);

2.4 注意

  • MyBatis一级缓存的生命周期和SqlSession一致。
  • MyBatis一级缓存内部设计简单,只是一个没有容量限定的HashMap,在缓存的功能性上有所欠缺。
  • MyBatis的一级缓存最大范围是SqlSession内部,有多个SqlSession或者分布式的环境下,数据库写操作会引起脏数据。
  • mybatis和spring整合后进行mapper代理开发,不支持一级缓存。

转自MyBatis的缓存机制

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值