MyBatis(4)一级缓存,二级缓存

MyBatis的缓存,包括一级缓存,二级缓存。

什么是一级缓存?

一级缓存指的就是sqlsession,在sqlsession中有一个数据区域,是map结构,这个区域就是一级缓存区域。一级缓存中的key是由sql语句、条件、statement等信息组成一个唯一值。一级缓存中的value,就是查询出的结果对象。

什么是二级缓存?

二级缓存指的就是同一个namespace下的mapper,二级缓存中,也有一个map结构,这个区域就是一级缓存区域。一级缓存中的key是由sql语句、条件、statement等信息组成一个唯一值。一级缓存中的value,就是查询出的结果对象。

一级缓存是默认使用的。

二级缓存需要手动开启。

MyBatis缓存框架图

MyBatis缓存框架图

一级缓存原理图

代码测试:

 /**
 * 测试一级缓存
 * @throws Exception
 */
@Test
public void testSelectStudentById() throws Exception {
    SqlSession session1 = sqlSessionFactory.openSession();
    // 拿到代理对象
    StudentMapper mapper = session1.getMapper(StudentMapper.class);
    //第一次查询id为1的Student的信息
    Student student1 = mapper.selectStudentById(1);
    System.out.println(student1);
    //第二次查询时,发现缓存中有id为1的Student的信息,于是从缓存中读取
    Student student2 = mapper.selectStudentById(1);
    System.out.println(student2);
    session1.close();
}

然后debug看一下

再回头看一下上面一级缓存原理图中间那一部分,当修改删除更新时commit,会自动清空缓存

代码测试:

 /**
 * 测试一级缓存
 * @throws Exception
 */
@Test
public void testSelectStudentById() throws Exception {
    SqlSession session1 = sqlSessionFactory.openSession();
    // 拿到代理对象
    StudentMapper mapper = session1.getMapper(StudentMapper.class);
    //第一次查询id为1的Student的信息
    Student student1 = mapper.selectStudentById(1);
    System.out.println(student1);
    //更新操作
    student1.setName("RonaldoWang");
    mapper.updateStudent(student1);
    //commit()就会清空缓存
    session1.commit();
    //第二次查询时,发现缓存中有id为1的Student的信息,于是从缓存中
    Student student2 = mapper.selectStudentById(1);
    System.out.println(student2);
    session1.close();
}

测试结果:


二级缓存原理图

看起来二级缓存与一级缓存相似,只是二级缓存的范围更广,区域划分是namespace,而一级缓存是在sqlsession里面。

1.开启二级缓存

第一步:总开关开启,需要在sqlMapConfig.xml中通过settings标签开启,默认它就是开启的,但还是写上去比较好,容易阅读吧。

第二步:在Mapper.xml下namespace中开启自己的开关

第三步:在PO类中实现序列化操作。

)

之所以需要实现序列化接口,是因为方便反序列化,二级缓存的区域不一定只是在内存中,也有可能在硬盘中。

测试代码:

/**
 * 测试二级缓存
 * @throws Exception
 */
@Test
public void testSelectStudentById() throws Exception {
    SqlSession session1 = sqlSessionFactory.openSession();
    SqlSession session2 = sqlSessionFactory.openSession();
    SqlSession session3 = sqlSessionFactory.openSession();

    StudentMapper mapper1 = session1.getMapper(StudentMapper.class);
    StudentMapper mapper2 = session2.getMapper(StudentMapper.class);
    StudentMapper mapper3 = session2.getMapper(StudentMapper.class);

    Student student1 = mapper1.selectStudentById(1);
    System.out.println(student1);
    //这里close很关键,在二级缓存中,close()会将数据放到二级缓存中
    session1.close();
    Student student3 = mapper3.selectStudentById(1);
    System.out.println(student3);
    session3.close();
}

测试结果:

上面代码中session2,和mapper2还没有用到,这里可以用mapper2给数据更新一下,看看是否会将缓存清空?

测试代码

/**
 * 测试二级缓存
 * @throws Exception
 */
@Test
public void testSelectStudentById() throws Exception {
    SqlSession session1 = sqlSessionFactory.openSession();
    SqlSession session2 = sqlSessionFactory.openSession();
    SqlSession session3 = sqlSessionFactory.openSession();

    StudentMapper mapper1 = session1.getMapper(StudentMapper.class);
    StudentMapper mapper2 = session2.getMapper(StudentMapper.class);
    StudentMapper mapper3 = session3.getMapper(StudentMapper.class);

    Student student1 = mapper1.selectStudentById(1);
    System.out.println(student1);
    //这里close很关键,在二级缓存中,close()会将数据放到二级缓存中
    session1.close();


    Student student2 = mapper2.selectStudentById(1);
    student2.setName("小六子");
    mapper2.updateStudent(student2);
    session2.commit();
    session2.close();


    Student student3 = mapper3.selectStudentById(1);
    System.out.println(student3);
    session3.close();
}

测试结果:


圈出来的看看就能理解了。

禁用缓存

)

刷新缓存

刷新缓存在select语句中默认是false

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值