走进Mybatis redis二级缓存 懒加载

发布于:2021-08-03 19:09:08



目录
背景Mybatis的一级缓存(默认开启)Mybatis的二级缓存二级缓存开启方式二级缓存存在的问题


Mybatis使用redis缓存自定义Mybatis二级缓存的实现类使用自定义缓存实现类


Mybatis延迟加载



MyBatis 是一款优秀的持久层框架,它支持定制化 SQL、存储过程以及高级映射。MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。MyBatis 可以使用简单的 XML 或注解来配置和映射原生信息,将接口和 Java 的 POJOs(Plain Ordinary Java Object,普通的 Java对象)映射成数据库中的记录。



背景

在整个系统运行过程中,最耗时的操作就是直接与计算机硬盘进行读写操作,尤其在数据库查询操作。这时缓存就显得十分重要,缓存主要解决高性能高并发的需求,也就是说对于一些需要复杂操作耗时查出来的结果,且确定后面不怎么变化,但是有很多读请求,那么直接将查询出来的结果放在缓存中,后面直接读缓存即可。


Mybatis的一级缓存(默认开启)

一级缓存,又叫本地缓存,是PerpetualCache类型的永久缓存,保存在执行器中(BaseExecutor),而执行器又在SqlSession(DefaultSqlSession)中,所以一级缓存的生命周期与SqlSession是相同的。



    Mybatis的一级缓存存放在SqlSession的生命周期,在同一个SqlSession中查询时,Mybatis会把执行的方法和参数通过算法生成缓存的键值,将键值和查询结果存入一个Map对象中。如果同一个SqlSession中执行的方法和参数完全一致,那么通过算法会生成相同的键值,当Map缓存对象中已经存在改键值时,则会返回缓存中的对象。(一个SqlSession连续两次查询 得到的是同一个java对象)任何的DML语句【insert update delete操作】都会清空一级缓存(增删改任何记录都会清空当前SqlSession的缓存)。

Mybatis的二级缓存

二级缓存,又叫自定义缓存,实现了Cache接口的类都可以作为二级缓存,所以可配置如encache等的第三方缓存。二级缓存以namespace名称空间为其唯一标识,被保存在Configuration核心配置对象中。



    MyBatis 一级缓存最大的共享范围就是一个SqlSession内部,那么如果多个SqlSession 需要共享缓存,则需要开启二级缓存,开启二级缓存后,会使用 CachingExecutor 装饰 Executor,进入一级缓存的查询流程前,先在CachingExecutor 进行二级缓存的查询。当二级缓存开启后,同一个命名空间(namespace) 所有的操作语句,都影响着一个 共同的 cache,也就是二级缓存被多个 SqlSession 共享,是一个全局的变量。当开启缓存后,数据的查询执行的流程就是 二级缓存 -> 一级缓存 -> 数据库

二级缓存开启方式

mybatis-config.xml配置文件通过添加以下内容来开启二级缓存,还需要在 Mapper 的xml 配置文件中加入 标签







特别注意:实现二级缓存的时候,MyBatis要求返回的POJO必须是可序列化的。


二级缓存存在的问题

默认使用Mybatis的二级缓存是单服务器工作,无法实现分布式缓存即不能实现多服务器信息共享。
解决方法:使用redis缓存实现分布式缓存(redis集群、主从复制)


Mybatis使用redis缓存

首先准备好(懒汉式获取)


    关于Jedis操作的工具类自定义序列化和反序列化工具类自定义工具类,获取SqlSessionFactory

自定义Mybatis二级缓存的实现类
    Mybatis二级缓存的实现类,主要实现使用redis内存数据库Cache缓存接口,供我们实现,然后重写方法,填写选用缓存技术是谁 Cache为缓存接口,给缓存供应商的SPI(Service Provider Interface)Cache接口的实现类必须有一个具有String类型参数的构造方法, 该参数作为实现类对象的id,对其进行唯一标识

public class MybatisCache implements Cache {

private String id;

public MybatisCache(String id) {
this.id = id;
}

/**
* 获取缓存对象的唯一标识
*/
@Override
public String getId() {
return this.id;
}

/**
* 保存key/value到缓存对象中
* key可以是任何对象
*/
@Override
public void putObject(Object key, Object value) {
JedisUtils.set(key, value);
}

/**
* 从缓存对象中获取key对应的value
*/
@Override
public Object getObject(Object key) {
return JedisUtils.get(key);
}

/**
* 可选的方法,移除key对应的value,没有被核心框架调用
*/
@Override
public Object removeObject(Object key) {
return null;
}

/**
* 清空缓存
*/
@Override
public void clear() {
JedisUtils.clear();
}

/**
* 获取缓存对象中存储的键/值对的数量
* 可选的方法,没有被框架核心调用
*/
@Override
public int getSize() {
return JedisUtils.getDBSize().intValue();
}

/**
* 重新equals方法
*/
@Override
public boolean equals(Object o) {
if (getId() == null)
throw new CacheException("Cache instances require an ID.");
if (this == o)
return true;
if (!(o instanceof Cache))
return false;

Cache otherCache = (Cache) o;
return getId().equals(otherCache.getId());
}

/**
* 重新hashCode方法
*/
@Override
public int hashCode() {
if (getId() == null)
throw new CacheException("Cache instances require an ID.");
return getId().hashCode();
}
}

使用自定义缓存实现类

Mapper 的xml 配置文件中 标签通过type属性指定缓存实现类




接下来只需要为需要缓存的查询语句添加flushCache和userCache属性





这样当第一次调用findAllUsers查询时,访问数据库将查询结果返回,并保存到redis数据库中。


第二次调用会直接从redis缓存中获取,大大提高查询效率。


Mybatis延迟加载

延迟加载就是懒加载,先去查询主表信息,如果用到从表的数据的话,再去查询从表的信息,也就是如果没用到从表的数据的话,就不查询从表的信息。



    mybatis-config.xml配置文件通过添加以下内容来开启懒加载








    编写测试代码

    测试吧!!!准备工作已经完毕

    由于开启懒加载,但是并没有使用user对象的方法,不会使用sql语句查询出user,因此显示为null
    修改代码,调用user的内容

    此时,会发出sql语句查询user
    因此,user就拥有了值!!!是不是很神奇?快动手试试吧


谢谢你能观看到这里,觉得对你有帮助,积极点赞
你的支持是我前进最大的动力!!!

相关推荐

最新更新

猜你喜欢