Mybatis的缓存分为一级缓存和二级缓存
1. 一级缓存
一级缓存是SqlSession级别的缓存,在操作数据库时需要构造sqlSession对象,在对象中有一个内存区,数据结构为(HashMap)用于存储缓存数据,不同的sqlSession之间
的缓存数据区域(HashMap)是互不影响的。
一级缓存的作用域是同一个SqlSession,在同一个SqlSession中如果执行两次相同的Sql语句,第一次查询的数据会缓存到缓存中,第二次将不会从数据库中查询。
Mybatis中一级缓存是开启的.
随便选择上面的一个查询语句,如
private static void selectByName(){
SqlSession session=DBTools.getSession();
PersonMapper mapper=session.getMapper(PersonMapper.class);
List<PersonBean> personList = mapper.selectByName("张三");
System.out.println(personList.toString());
}
执行后看日志:
如果在函数中使用此session再查一次的话,即:
// 一级缓存
private static void oneLevelCache1(){
SqlSession session=DBTools.getSession();
PersonMapper mapper=session.getMapper(PersonMapper.class);
List<PersonBean> personList = mapper.selectByName("张三");
System.out.println(personList.toString());
System.out.println("======================");
List<PersonBean> personList2 = mapper.selectByName("张三");
System.out.println(personList2.toString());
}
可以看到第二次查询没有输出查询语句,说明没有去查询数据库,直接从内存中读取了值。
如果不希望使用缓存数据,可以使用clearCache清空缓存。
session.clearCache();
另一个例子,使用两个不同的session:
// 测试一级缓存
private static void oneLevelCache2(){
SqlSession session=DBTools.getSession();
PersonMapper mapper=session.getMapper(PersonMapper.class);
List<PersonBean> personList = mapper.selectByName("张三");
System.out.println(personList.toString());
System.out.println("======================");
SqlSession session2=DBTools.getSession();
PersonMapper mapper2=session2.getMapper(PersonMapper.class);
List<PersonBean> personList2 = mapper2.selectByName("张三");
System.out.println(personList2.toString());
}
也会查两次数据库。
2. 二级缓存
二级缓存默认是关闭的。
二级缓存与一级缓存的区别,二级缓存的范围更大,多个SqlSession可以共享
Mapper中的数据
每一个namespace的mapper都有一个二级缓存的区域,如果两个namespace相同,则这两个mapper执行的sql将放在相同的缓存中。
一级缓存来自于同一个session:
SqlSession session=sessionFactory.openSession();
二级缓存来自于同一个namespace,即sessionFactory对应的"mybatis.cfg.xml"
中注册了许多的xml,每个xml都是一个mapper,对应一个namespace:
<mapper namespace="cn.harvetech.mapper.WmFeedbackMapper" >
开启二级缓存的方法很简单,在mapper标签下添加标签:
<cache></cache>
测试:还用上面的例子:
// 测试一级缓存
private static void oneLevelCache2(){
SqlSession session=DBTools.getSession();
PersonMapper mapper=session.getMapper(PersonMapper.class);
List<PersonBean> personList = mapper.selectByName("张三");
System.out.println(personList.toString());
session.commit();
System.out.println("======================");
SqlSession session2=DBTools.getSession();
PersonMapper mapper2=session2.getMapper(PersonMapper.class);
List<PersonBean> personList2 = mapper2.selectByName("张三");
System.out.println(personList2.toString());
}
但是需要注意,使用完一个session的方法后需要提交一下,才能被缓存到二级缓存中。
或者调用session.close();方法
还需要注意一个点,使用二级缓存的话,实体类需要能序列化,即:implements Serializable
注:如果不行的话可能需要在mybatis.cfg.xml中开始位置添加如下配置:
<settings>
<!--开启二级缓存-->
<setting name="cacheEnabled" value="true"/>
</settings>