Mybatis系列--16-缓存

2022/7/25 23:25:41

本文主要是介绍Mybatis系列--16-缓存,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

Mybatis系列--16-缓存

概述

  1. 什么是缓存?
    存在内存中的临时数据
    将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系数据库)中查询,直接从缓存中查询,提高了查询效率,解决了高并发系统的性能问题

  2. 为什么使用缓存?
    减少和数据库的交互次数,减少数据库的压力,减少系统开销,提高系统效率

  3. 什么样的数据能使用缓存?
    经常查询并且不经常改变的数据

  4. Mybatis缓存
    Mybatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大的提升查询效率
    Mybatis系统中默认定义了两级缓存,一级缓存和二级缓存
    默认情况下,只有一级缓存开启(sqlsession级别的环迅,也称为本地缓存)
    二级缓存需要手动开启和配置,他是基于namespace级别的缓存
    为了提高扩展性,Mybatis定义了缓存扩展接口Cache。我们可以通过实现Cache接口来自定义二级缓存

一级缓存

一级缓存即SqlSession级别的缓存,在一个SqlSession被建立和关闭之前生效,一级缓存是默认开启的且无法关闭
所有的查询语句的执行都会被缓存
如以下测试代码

查询缓存的例子

import org.apache.ibatis.session.SqlSession;
import org.junit.Test;

import com.kuang.dao.UserMapper;
import com.kuang.pojo.User;
import com.kuang.util.MyBatisUtil;

/**
 * 功能描述
 *
 * @since 2022-07-25
 */
public class MyTest {
    @Test
    public void testCache() {
        SqlSession session = MyBatisUtil.getSqlsession();
        UserMapper mapper = session.getMapper(UserMapper.class);
        User user = mapper.getUserById(1);
        System.out.println(user);
        System.out.println("======================");
        User user1 = mapper.getUserById(1);
        System.out.println(user1);
        session.close();
    }
}

开启日志的情况下可以看到查询结果

Created connection 988800485.
==>  Preparing: select * from `user` where id = ?
==> Parameters: 1(Integer)
<==    Columns: id, name, pwd
<==        Row: 1, 狂神, 123456
<==      Total: 1
User(id=1, name=狂神, pwd=123456)
======================
User(id=1, name=狂神, pwd=123456)
Closing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@3aefe5e5]
Returned connection 988800485 to pool.

可以看到sql的执行仅执行了一次

以下情况下不会查询缓存的信息

  1. 第一次查询1号用户,第二次查询2号用户,查询2号的时候不会查询到缓存中去,因为2号用户的信息还没有缓存
  2. 在两次查询1号用户之间对数据库执行了增删改的操作,则之前的缓存会失效
  3. 使用session.clearCache()代码手动清除缓存
  4. 查询不同的Mapper.xml文件

一级缓存就是一个map,查询完一个就往map中插入一条key,value

二级缓存

二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存
二级缓存是基于namespace级别的缓存,一个命名空间,对应一个二级缓存
工作机制
1.一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中
2.如果当前会话关闭了,这个会话对应的一级缓存就没有了,但是会话关闭后,一级缓存中的数据会被保存到二级缓存中
3.新的会话查询信息,基于可以从二级缓存中去获取
4.不同的mapper查出来的数据会放在自己对应的缓存(map)中

测试

1.开启全局缓存

<!-- 显示开启缓存,其实不写也是默认开启的 -->
<setting name="cacheEnabled" value="true" />

2.开启二级缓存

<cache/>

或者

<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true" />

测试代码

import org.apache.ibatis.session.SqlSession;
import org.junit.Test;

import com.kuang.dao.UserMapper;
import com.kuang.pojo.User;
import com.kuang.util.MyBatisUtil;

/**
 * 功能描述
 *
 * @since 2022-07-25
 */
public class MyTest {

    @Test
    public void testCache2() {
        SqlSession session = MyBatisUtil.getSqlsession();
        UserMapper mapper = session.getMapper(UserMapper.class);
        User user = mapper.getUserById(1);
        System.out.println(user);
        session.close();
        SqlSession session1 = MyBatisUtil.getSqlsession();
        UserMapper mapper1 = session1.getMapper(UserMapper.class);
        User user1 = mapper1.getUserById(1);
        System.out.println(user1);
        session.close();
        System.out.println(user == user1);

    }
}


这篇关于Mybatis系列--16-缓存的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!


扫一扫关注最新编程教程