多学习。

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

posted @   czyaaa  阅读(33)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
点击右上角即可分享
微信分享提示