一文彻底搞懂Mybatis系列(十四)之MyBatis一级缓存

简介: 一文彻底搞懂Mybatis系列(十四)之MyBatis一级缓存

一、MyBatis的缓存

1、缓存的作用

通过减少10的方式,来提高程序的执行效率。

2、mybatis的缓存

将select语句的查询结果放到缓存(内存)当中,下一次还是这条select语句的话,直接从缓存中取,不再查数据库。

一方面是减少了IO。

另一方面不再执行繁琐的查找算法。效率大大提升。

3、mybatis缓存包括

  • 一级缓存:将查询到的数据存储到SqlSession中。
  • 二级缓存:将查询到的数据存储到SqlSessionFactory中。
  • 或者集成其它第三方的缓存:比如EhCache【Java语言开发的】、Memcache【C语言开发的】等。

4、注意点

缓存只针对于DQL语句,也就是说缓存机制只对应select语句

二、一级缓存

提醒:

一级缓存默认是开启的,不需要做任何配置。

原理:

只要使用同一个SqlSession对象执行同一条SQL语句,就会走缓存。

下面通过代码来进行演示

pojo类 Clazz

public class Clazz {
    private Integer cid;
    private String name;
    ......此处省略get、set等方法
}

ClazzMapper接口

public interface ClazzMapper {
    public Clazz selectByCidStep2(Integer cid);
    int insertClazz(Clazz clazz);
}

ClazzMapper.xml文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.powernode.mybatis.mapper.ClazzMapper">
    <select id="selectByCidStep2" resultType="Clazz">
        select * from t_clazz where cid = #{cid}
    </select>
    <insert id="insertClazz" parameterType="Clazz">
        insert into t_clazz values (#{cid},#{name})
    </insert>
</mapper>

测试类趴一下

@Test
    public void testcache() throws ParseException {
        SqlSession sqlSession = SqlSessionUtil.openSqlSession();
        ClazzMapper mapper = sqlSession.getMapper(ClazzMapper.class);
        Clazz clazz = mapper.selectByCidStep2(1000);
        System.out.println(clazz);
        ClazzMapper mapper1 = sqlSession.getMapper(ClazzMapper.class);
        Clazz clazz1 = mapper1.selectByCidStep2(1000);
        System.out.println(clazz1);
        sqlSession.close();
    }

运行结果

ClazzMapper.selectByCidStep2 - ==>  Preparing: select * from t_clazz where cid = ?
ClazzMapper.selectByCidStep2 - ==> Parameters: 1000(Integer)
ClazzMapper.selectByCidStep2 - <==      Total: 1
Clazz{cid=1000, name='高三一班', stus=null}
Clazz{cid=1000, name='高三一班', stus=null}

我们从运行结果发现,只执行了一次sql语句的查询,第二次查询,是直接从一级缓存中拿的数据,并没有执行sql。所以说mybatis默认是开启一级缓存的,但是前提必须是同一个Sqlsession对象。

三、关于一级缓存的思考

1、思考:什么时候不走缓存?

SqlSession对象不是同一个,肯定不走缓存。查询条件不一样,肯定也不走缓存。

代码演示:

注意:接口、pojo类都是基于上面的哦

我们使用两个不同的SqlSession对象来执行sql语句,看看结果如何。

@Test
    public void testcache1() throws Exception {
        SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml"));
        SqlSession sqlSession1 = factory.openSession();
        ClazzMapper mapper = sqlSession1.getMapper(ClazzMapper.class);
        Clazz clazz = mapper.selectByCidStep2(1000);
        System.out.println(clazz);
        SqlSession sqlSession2 = factory.openSession();
        ClazzMapper mapper1 = sqlSession2.getMapper(ClazzMapper.class);
        Clazz clazz1 = mapper1.selectByCidStep2(1000);
        System.out.println(clazz1);
        sqlSession1.commit();
        sqlSession2.close();
    }

运行结果如下,执行了两次相同的sql语句,因为他们使用的是不同的SqlSession对象。

2、思考:什么时候一级缓存失效?

第一次DQL和第二次DQL之间你做了以下两件事中的任意一件,都会让一级缓存清空:

  • 执行了sqlSession的clearCacheO)方法,这是手动清空缓存。
  • 执行了INSERT或DELETE或UPDATE语句。不管你是操作哪张表的,都会清空一级缓存。

代码演示:

注意:接口、pojo类都是基于上面的哦

我们先演示调用了sqlSession的clearCacheO)方法,看看结果如何

@Test
    public void testClearcache() throws ParseException {
        SqlSession sqlSession = SqlSessionUtil.openSqlSession();
        ClazzMapper mapper = sqlSession.getMapper(ClazzMapper.class);
        Clazz clazz = mapper.selectByCidStep2(1000);
        System.out.println(clazz);
        sqlSession.clearCache();
        ClazzMapper mapper1 = sqlSession.getMapper(ClazzMapper.class);
        Clazz clazz1 = mapper1.selectByCidStep2(1000);
        System.out.println(clazz1);
        sqlSession.close();
    }

运行结果如下,执行了两次同样的sql查询,说明调用了clearCache方法之后,一级缓存失效了

再演示一下,在两次查询之间,添加一个插入的操作,看看结果如何

@Test
    public void testNocache() throws ParseException {
        SqlSession sqlSession = SqlSessionUtil.openSqlSession();
        ClazzMapper mapper = sqlSession.getMapper(ClazzMapper.class);
        Clazz clazz = mapper.selectByCidStep2(1000);
        System.out.println(clazz);
        mapper.insertClazz(new Clazz(1004,"高三一班"));
        sqlSession.commit();
        ClazzMapper mapper1 = sqlSession.getMapper(ClazzMapper.class);
        Clazz clazz1 = mapper1.selectByCidStep2(1000);
        System.out.println(clazz1);
        sqlSession.close();
    }

运行结果如下,在两次查询之间增加INSERT或DELETE或UPDATE语句,不管你是操作哪张表的,都会清空一级缓存。


相关文章
|
6月前
|
缓存 Java 数据库连接
Mybatis缓存相关面试题有多卷
使用 MyBatis 缓存机制需要注意以下几点: 对于频繁更新和变动的数据,不适合使用缓存。 对于数据的一致性要求比较高的场景,不适合使用缓存。 如果配置了二级缓存,需要确保缓存的数据不会影响到其他业务模块的数据。 在使用缓存时,需要注意缓存的命中率和缓存的过期策略,避免缓存过期导致查询性能下降。
105 0
|
2月前
|
缓存 Java 数据库连接
mybatis复习05,mybatis的缓存机制(一级缓存和二级缓存及第三方缓存)
文章介绍了MyBatis的缓存机制,包括一级缓存和二级缓存的配置和使用,以及如何整合第三方缓存EHCache。详细解释了一级缓存的生命周期、二级缓存的开启条件和配置属性,以及如何通过ehcache.xml配置文件和logback.xml日志配置文件来实现EHCache的整合。
mybatis复习05,mybatis的缓存机制(一级缓存和二级缓存及第三方缓存)
|
20天前
|
SQL 缓存 Java
【详细实用のMyBatis教程】获取参数值和结果的各种情况、自定义映射、动态SQL、多级缓存、逆向工程、分页插件
本文详细介绍了MyBatis的各种常见用法MyBatis多级缓存、逆向工程、分页插件 包括获取参数值和结果的各种情况、自定义映射resultMap、动态SQL
【详细实用のMyBatis教程】获取参数值和结果的各种情况、自定义映射、动态SQL、多级缓存、逆向工程、分页插件
|
21天前
|
SQL 缓存 Java
MyBatis如何关闭一级缓存(分注解和xml两种方式)
MyBatis如何关闭一级缓存(分注解和xml两种方式)
61 5
|
5月前
|
SQL 缓存 Java
MYBATIS缓存
MYBATIS缓存
|
1月前
|
缓存 Java 数据库连接
使用MyBatis缓存的简单案例
MyBatis 是一种流行的持久层框架,支持自定义 SQL 执行、映射及复杂查询。本文介绍了如何在 Spring Boot 项目中集成 MyBatis 并实现一级和二级缓存,以提高查询性能,减少数据库访问。通过具体的电商系统案例,详细讲解了项目搭建、缓存配置、实体类创建、Mapper 编写、Service 层实现及缓存测试等步骤。
|
4月前
|
SQL 缓存 Java
【面试官】Mybatis缓存有什么问题吗?
面试官:你说下对MyBatis的理解?面试官:那SqlSession知道吧?面试官:Mybatis的缓存有哪几种?面试官:那Mybatis缓存有什么问题吗?面试官:Mybatis分页插件是怎么
【面试官】Mybatis缓存有什么问题吗?
|
4月前
|
缓存 算法 Java
关于MyBatis的缓存详解
MyBatis 的缓存机制非常灵活,可以通过简单的配置来满足不同的性能需求。合理地使用缓存可以显著提高应用程序的性能,尤其是在处理大量数据库查询时。然而,开发者需要注意缓存的一致性和并发问题,特别是在使用可读写缓存时。
|
5月前
|
缓存 NoSQL Java
在 SSM 架构(Spring + SpringMVC + MyBatis)中,可以通过 Spring 的注解式缓存来实现 Redis 缓存功能
【6月更文挑战第18天】在SSM(Spring+SpringMVC+MyBatis)中集成Redis缓存,涉及以下步骤:添加Spring Boot的`spring-boot-starter-data-redis`依赖;配置Redis连接池(如JedisPoolConfig)和连接工厂;在Service层使用`@Cacheable`注解标记缓存方法,指定缓存名和键生成策略;最后,在主配置类启用缓存注解。通过这些步骤,可以利用Spring的注解实现Redis缓存。
79 2
|
5月前
|
SQL 缓存 Java
Java框架之MyBatis 07-动态SQL-缓存机制-逆向工程-分页插件
Java框架之MyBatis 07-动态SQL-缓存机制-逆向工程-分页插件