一、什么是缓存
在内存中存储的临时数据。
二、为什么使用缓存
减少和数据库的交互次数,提高代码的执行效率。
三、什么样的数据能使用缓存,什么样的数据不能使用?
代码语言:javascript复制 适用于缓存:
需要经常查询且不经常修改的。
数据的正确与否对最终结果影响不大的。
不适用于缓存:
经常改变的数据(银行利率等)
数据的正确与否对最终结果影响很大的。
四、Mybatis中的一级缓存和二级缓存
一级缓存: 它指的是Mybatis中SqlSession对象的缓存。 当我们执行查询之后,查询的结果会同时存入到SqlSession为我们提供一块区域中。 该区域的结构是一个Map。当我们再次查询同样的数据,mybatis会先去sqlsession中 查询是否有,有的话直接拿出来用。 当SqlSession对象消失时,mybatis的一级缓存也就消失了。
二级缓存: 它指的是Mybatis中SqlSessionFactory对象的缓存。由同一个SqlSessionFactory对象创建的SqlSession共享其缓存。
五、Mybatis中的一级缓存
新建一个Maven工程 1、pojo类
代码语言:javascript复制 private Integer id;
private String username;
private String password;
private String sex;
2、dao层接口
代码语言:javascript复制 User findById(Integer id);
3、映射文件
代码语言:javascript复制<mapper namespace="com.rpf.dao.IUserDao">
<!--配置查询所有-->
<select id="findById" parameterType="int" resultType="user" useCache="true">
select * from user where id=#{uid}
</select>
</mapper>
4、测试
代码语言:javascript复制 @Test
public void testLevelCache() {
User u1 = userDao.findById(41);
System.out.println(u1);
User u2 = userDao.findById(41);
System.out.println(u2);
System.out.println(user1 == user2);
}
5、结果:
只执行了一条查询语句,第二次直接调用缓存中的结果
六、触发一级缓存清空情况
此时如果加入调用一个更新方法更新数据
代码语言:javascript复制 @Test
public void testClearCache() {
User u1 = userDao.findById(42);
System.out.println(u1);
user1.setUsername("clean Cache");
user1.setAddress("中国");
userDao.updateUser(u1);
User u2 = userDao.findById(42);
System.out.println(u2);
System.out.println(u1 == u2);
}
结果不相等
查询了两次
结论:当调用sqlSession的删除、修改、添加、commit()、close()方法时,就会清空一级缓存。
七、Mybatis的二级缓存
二级缓存的使用步骤: 第一步:让Mybatis框架支持二级缓存(在SqlMapConfig.xml中配置)
代码语言:javascript复制 <settings>
<setting name="cacheEnabled" value="true"/>
</settings>
<!--默认开启-->
第二步:让当前的映射文件支持二级缓存(在IUserDao.xml中配置)
代码语言:javascript复制<!-- 开启User支持二级缓存-->
<cache/>
第三步:让当前的操作支持二级缓存(在select标签中配置)
代码语言:javascript复制<!---useCache标签-->
<select id="findById" parameterType="int" resultType="user" useCache="true">
select * from user where id=#{uid}
</select>
第四步:测试
代码语言:javascript复制@Test
public void testCache(){
SqlSession sqlSession1 = factory.openSession();
UserDao dao1 = sqlSession1.getMapper(UserDao.class);
User 1 = dao1.findById(11);
System.out.println(u1);
sqlSession1.close();//一级缓存消失
SqlSession sqlSession2 = factory.openSession();//新建一个缓存对象
UserDao dao2 = sqlSession2.getMapper(UserDao.class);
User u2 = dao2.findById(11);
System.out.println(u2);
sqlSession2.close();//一级缓存消失
System.out.println(user1 == user2);
}
两个对象不是同一个,但只查询了一次 说明二级缓存存在。
两次对象不相同是因为在二级缓存中,存储的内容是数据而不是对象 当一级缓存消失后再次访问这个数据时 它会创建一个新的对象把值赋给它。所以两次对象不一样,但只执行了一次查询语句,因为二级缓存存在。