当前位置: 代码迷 >> 综合 >> MyBatis一级缓存、二级缓存、禁用清理二级缓存的使用
  详细解决方案

MyBatis一级缓存、二级缓存、禁用清理二级缓存的使用

热度:21   发布时间:2024-01-31 21:22:01.0

一级缓存

作用范围在同一个SqlSession对象中,Mybatis自动打开了一级缓存,缓存保存在内存中,当进行一次sql查询时,Mybatis会将查询结果放入一级缓存中,如果需要再进行一个同样的查询,Mybatis则会直接从缓存中获取上次查询的结果,从而减少数据库的访问,从而增加性能。这就导致了如果在第二次查询之前,进行了增删改操作,那么第二次查询的结果则是不准确的,所以需要我们在每次增删改之后使用一次sqlsession.commit()方法,该方法会将一级缓存清空,保证第二次查询的结果准确。

二级缓存

作用范围是同一个namepace,即只要是同一个xml内的namepace里的方法,都共享二级缓存,Mybatis默认未打开二级缓存,需要手动设置打开。并且,二级缓存存放在硬盘中而非内存中,所以涉及到的所有实体类需要序列化(将缓存变为硬盘数据),序列化非常简单,仅需在实体类上继承一个接口Serializable,但是,实体类的子类父类均需要序列化。
数据首先存放在一级缓存中,当SqlSesision.close()执行时,才会将数据放入二级缓存中,这种特性可以减少IO操作,提升性能。
注意:如果多个xxxMapper.xml共用一个namepace,则他们仍然共享一个二级缓存。
conf.xml配置:打开二级缓存。

<settings><!-- 开启二级缓存 --><setting name="cacheEnabled" value="true" />
</settings>

mapper.xml配置:声明此mapper开启二级缓存。

<cache/>

对应的实体类继承序列化接口

public class Class implements Serializable {Student student =new Student();//Class类内的子类也需要序列化...
}
public class Student implements Serializable {...
}

禁用某个查询的二级缓存

添加useCache=“flase”>

  <select id="updatePersonById"  parameterType="person" useCache="flase">...</select>

清理二级缓存

1:commit()

二级缓存的清理与一级缓存一致,执行sqlsession.commit()后即被清空,但是,sqlsession.commit()中的sqlsession不可以是查询语句的!在某个询语句中,使用commit无法清理二级缓存。例如:

	public static void queryPerson() throws IOException {//加载MyBatis配置文件Reader reader = Resources.getResourceAsReader("conf.xml");SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(reader);//session 相当于 JDBC的 connection//查询sessionSqlSession selectSession = sessionFactory.openSession();PersonMapper personMapper = selectSession.getMapper(PersonMapper.class);...personMapper.commit()//此conmmit无效//修改sessionSqlSession updateSession = sessionFactory.openSession();PersonMapper personMapper = updateSession.getMapper(PersonMapper.class);...updateSession.commit()//此conmmit有效session.close();}
2:使用flushCache=“true”

在select标签后添加flushCache=“true”。此语当用在select语句时,表示调用select语句不会清空本地缓存和二级缓存,用在insert,update.delete时,会导致清空本地和二级缓存。

  相关解决方案