Mybatis中的缓存
引言
什么是缓存:
存在于内存中的临时数据。
为什么使用缓存:
减少和数据库的交互次数,提高执行效率。
什么样的数据能用缓存,什么数据不能使用:
适用于缓存:
经常查询并且不经常改变的
数据的正确与否对最终结果影响不大(因为缓存代表着会与数据库数据不同)
不适用于缓存:
经常改变的数据
数据的正确与否对最终结果影响很大的
例如:商品的库存,银行的汇率,股市的牌价
一级缓存
定义
mybatis中SqlSession对象的缓存。当我们执行查询之后,查询的结果会同时存入到SqlSession为我们提供的一块区域中。该区域的结构是一个Map。当我们再次查询同样的数据,mybatis会先去SqlSession中查看是否有,有的话直接拿出来用。
当SqlSession对象消失时,一级缓存也就消失了。
一级缓存演示
@Test
public void testFirstLevelCache(){
User user1 = dao.findById(41);
System.out.println(user1);
User user2 = dao.findById(41);
System.out.println(user2);
System.out.println(user1 == user2);
}
只进行一次查询,第二次查询使用的是一级缓存,获取的对象也是相同地址的。
@Test
public void testFirstLevelCache(){
User user1 = dao.findById(41);
System.out.println(user1);
session.close();
session = factory.openSession();
dao = session.getMapper(UserDao.class);
/**可用:session.clearCache(); 清空缓存**/
User user2 = dao.findById(41);
System.out.println(user2);
System.out.println(user1 == user2);
}
执行了两次查询,对象地址不相同,证明第二个user是再new一次的
如果一级缓存中的数据域数据库不一致如何做到同步
更新办法
<update id="updateUser" parameterType="user">
update user set id = #{id}
<if test="username != null and username != ''">
,username = #{username}
</if>
<if test="address != null and address != ''">
,address = #{address}
</if>
<if test="sex != null and sex != ''">
,sex = #{sex}
</if>
<if test="birthday != null">
,birthday = #{birthday}
</if>
where id = #{id}
</update>
演示
@Test
public void testFirstLevelCache(){
User user1 = dao.findById(41);
System.out.println(user1);
User user = new User();
user.setId(41);
user.setUsername("clear cache");
user.setSex("女");
dao.updateUser(user);
User user2 = dao.findById(41);
System.out.println(user2);
System.out.println(user1 == user2);
}
可以看出更新后,就不会从一级缓存中查找,而是进行一次新的查询
原因
当调用SqlSession的修改,添加,删除,commit(),close()等方法时,就会清空一级缓存
二级缓存
定义
指的是Mybatis中SqlSessionFactory对象的缓存。由同一个SqlSessionFactory对象创建的SqlSession共享其缓存。
二级缓存的使用步骤
第一步:让mybatis框架支持二级缓存(在SqlMapConfig.xml中配置)
<settings>
<setting name="cacheEnabled" value="true"/>
</settings>
第二步:让当前的映射文件支持二级缓存(在UserDao.xml中配置)
<!--开启user支持二级缓存-->
<cache></cache>
@CacheNamespace(blocking = true)
public interface UserDao {
...
...
...
}
第三步:让当前的操作支持二级缓存(在select标签中配置)(注解配置不需要此步)
<!--根据id查询单个用户-->
<select id="findById" resultType="com.czy.domain.User" parameterType="INT" useCache="true">
select * from user where id = #{id};
</select>
测试
@Test
public void testSecondLevelCache(){
SqlSession session1 = factory.openSession();
UserDao dao1 = session1.getMapper(UserDao.class);
User user1 = dao1.findById(41);
System.out.println(user1);
session1.close(); //一级缓存消失
SqlSession session2 = factory.openSession();
UserDao dao2 = session2.getMapper(UserDao.class);
User user2 = dao2.findById(41);
System.out.println(user2);
session2.close();
System.out.println(user1 == user2);
}
只进行了一次查询,第二次直接调用缓存,由于存储的是数据不是对象,故比较结果为false
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!