mybatis之二级缓存
创始人
2025-05-31 23:02:41
0

安装 redis

安装redis windows版。

安装redis-desktop-manager-0.8.8.384.exe

redis 服务端

  1. 解压 安装包,选择指定目录(最好全英文目录)

选择 Redis-x64-3.2.100.zip,进行解压

  1. 部署 Redis在windows下的服务

  1. 管理员身份打开 命令提示符

  1. 切换路径为 解压安装的目录

  1. 命令行使:redis-server --service-install redis.windows.conf

部署完成:设置自启动(内存占用很小)

redis使用命令

服务

命令

卸载服务

redis-server --service-uninstall

开启服务

redis-server --service-start

停止服务

redis-server --service-stop

也可以进入 计算机管理,选择服务,鼠标右键启动或停止服务。

redis 客户端

  1. 选择 redis-desktop-manager-2020.1.0.0.exe1 2020版本的 redis客户端(中文的),双击运行

  1. 下一步,下一步,选择安装目录(最好全英文目录)

  1. 下载完成,询问是否 安装完成自动打开,勾选 Finish

主界面

  1. 配置 文件,开启远程访问

  1. 查看 所需的文件,Redis服务右键属性查看

  1. 在 安装目录找到对应的文件

  1. 修改 连接密码

搜索 requirepass 关键字,取消注释并 确定密码(顶格写!)

  1. 注释 本地连接对应的 bind 127.0.0.1

搜索 bind 关键字

  1. 将redis默认的守护关闭

搜索 protected 关键字

修改 protected-mode yes 为 protected-mode no

  1. 保存文件,重启 redis 服务生效

修改配置,重新启动服务才会有效

新建 redis连接

  • 新建 redis 连接,确定就可以建立了

  • 默认有 十六个数据库 0-15,不需要自己创

(如果需要可以手动添加)

redis 小知识

相关知识

  • redis 也是一个框架

  • 可以 抗住 高并发的压力,MySQL不行。

  • Redis 缓存数据具有时效性,过期会自动删除(3600s 为 一小时)

  • redis 集群:

  • 集群是 redis不仅只有一个节点,而是 有很多节点

  • 为什么会有集群??

  • 如果 redis只有一个节点,为单节点,它挂了,与之所有相关连接的客户端全部报错

  • 如果是集群,有多个节点,一个节点挂了,还有其他节点在工作,可以切换到其他节点进行连接

  • redis 只能存储 key,value值

  • 集合 或对象只能通过 Jackson转成 json对象(本质为 字符串),再把数据储存进 redis

  • 多端模式(手机端,电脑端,网页端...)

  • session 不合适,第三方存储 redis合适

  • cookie 也不合适,手机端不支持,小程序也不支持,redis支持

五大常用类型

详情参考 https://zhuanlan.zhihu.com/p/161311820

数据类型

使用场景

String 字符串类型

1.缓存结构体信息2.计数功能

List列表类型

1.异步队列2.秒杀抢购

Hash数据类型

1.保存结构体信息

Set集合类型

1.去重

Zset有序集合

1.各类热门排序

spring+redis集成配置

pom.xml 导入依赖

定义版本

1.7.1.RELEASE

2.9.0

2.9.3

导入依赖

redis.clients

jedis

${jedis.version}

commons-pool2

org.apache.commons

org.springframework.data

spring-data-redis

${spring-data-redis.version}

com.fasterxml.jackson.core

jackson-databind

${jackson.version}

jackson-annotations

com.fasterxml.jackson.core

com.fasterxml.jackson.core

jackson-core

${jackson.version}

com.fasterxml.jackson.core

jackson-annotations

${jackson.version}

新建 spring-redis.xml

新建 spring 与 redis 的集成文件

记得 在 spirng.xml 文件中引入 spring-redis.xml!!!

步骤

  1. 导入外部数据库连接 配置文件 redis.properties

  1. 创建数据库连接池

  1. 创建 Redis 连接工厂类 ConnectionFactory

  1. 创建 RedisTemplate 模板类

导入外部 Redis配置文件

转换编码格式

文件是 编码格式:(记得 检查ip地址和密码,不正确则修改)

idea 可 转换为 中文(utf-8)可视化读,但是源文件还是 编码格式

打开设置:File 选择 Settings

导入 redis.properties

多配置文件.properties文件引入方式:

class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">

classpath:jdbc.properties

classpath:redis.properties

将spring-mybatis.xml和spring-redis.xml中引入的properties配置文件注释,并统一到spring.xml文件中配置引入。

创建 Redis数据库连接池

创建 Redis连接工厂类

destroy-method="destroy">

创建 RedisTemplate模板类

这里针对两种类型

  • String类型

  • 添加序列化 (不进行序列化,redis缓存不了)

  • Hash类型

复杂类型需要 转换后再添加到 redi内

启用spring注解式缓存

配置Redis缓存管理器

配置redis缓存过期时间、是否使用缓存前缀及缓存前缀配置。

配置自定义Key生成器

bean注入

导入自定义Key生成器 CacheKeyGenerator

引入 CacheKeyGenerator类

packagecom.zking.ssm.util;

@Slf4j

/**

* 自定义 Key生成

*/

publicclassCacheKeyGeneratorimplementsKeyGenerator {

// custom cache key

publicstaticfinalintNO_PARAM_KEY=0;

publicstaticfinalintNULL_PARAM_KEY=53;

/**

* 重写方法:生成 Key规则

* @param target 目标对象

* @param method 目标方法

* @param params 目标方法的执行参数

* @return

*/

@Override

publicObjectgenerate(Objecttarget, Methodmethod, Object... params) {

//动态的 可变字符串

StringBuilderkey=newStringBuilder();

/* target.getClass().getSimpleName() = BookServiceImpl

1)-cache-BookServiceImpl.queryBookPager:0 无参

2)-cache-BookServiceImpl.queryBookPager:zs,10 ??啥情况

3)-cache-BookServiceImpl.queryBookPager:53 参数为 null值,赋值常量

4)-cache-BookServiceImpl.queryBookPager:89,90,91... 判断参数是否为集合或集合,循环遍历取出单个参数

5)-cache-BookServiceImpl.queryBookPager:zs String类型,直接拼接参数

6)-cache-BookServiceImpl.queryBookPager:109 HashCode值

*/

key.append(target.getClass().getSimpleName()).append(".").append(method.getName()).append(":");

if (params.length==0) {

key.append(NO_PARAM_KEY);

} else {

intcount=0;

for (Objectparam : params) {

if (0!=count) {//参数之间用,进行分隔

key.append(',');

}

if (param==null) {

key.append(NULL_PARAM_KEY);

} elseif (ClassUtils.isPrimitiveArray(param.getClass())) {

intlength=Array.getLength(param);

for (inti=0; i

key.append(Array.get(param, i));

key.append(',');

}

} elseif (ClassUtils.isPrimitiveOrWrapper(param.getClass()) ||paraminstanceofString) {

key.append(param);

} else {//Java一定要重写hashCode和eqauls

key.append(param.hashCode());

}

count++;

}

}

//根据规则生成的 key,转成字符串,最终输出 字符串类型数据

StringfinalKey=key.toString();

log.debug("using cache key={}", finalKey);

returnfinalKey;

}

}

启用注解式缓存

缓存注解

@CacheConfig

@CacheConfig是一个类级别的注解,允许共享缓存的名称、KeyGenerator、CacheManager和CacheResolver。

参数

说明

value

缓存位置的一段名称,不能为空

key

缓存的key,默认为空,表示使用方法的参数类型及参数值作为key,支持SpEL

@Cacheable

配置在方法或类上

作用:

本方法执行后,先去缓存看有没有数据,如果没有,从数据库中查找出来,给缓存中存一份,返回结果,次本方法执行,在缓存未过期情况下,先在缓存中查找,有的话直接返回,没有的话从数据库查找。

先在 redis缓存查,没有则先去数据库真实查询并备份在 redis中,有的则直接在 redis取出数据

属性

参数

说明

value

缓存位置的一段名称,不能为空

key

缓存的key,默认为空,表示使用方法的参数类型及参数值作为key,支持SpEL

keyGenerator

指定 key的生成策略(覆盖默认策略)

condition

触发条件,满足条件就加入缓存,默认为空,表示全部都加入缓存,支持SpEL

//'' 字符串,随意填,覆盖 默认策略
//#bookId SpEL表达式
@Cacheable(value="selectOne",key="'book-'+#bookId")

+ 只有一个 ` " "` ,默认是 value的属性值;如果有其他属性,则必须添加 `value=""`

+ value 补全在 缓存前缀的前面

+ 缓存的值就是 当前方法的返回值

在 接口实现类添加注解

示例:

@Cacheable(value="selectByPrimaryKey",key="'selectOne-'+#bookId",condition="#bookId>90")

@Override

publicBookselectByPrimaryKey(IntegerbookId) {

Bookbook=bookMapper.selectByPrimaryKey(bookId);

returnbook;

}

redis 缓存

@CachePut

作用

更新操作,即每次不管缓存中有没有结果,都从数据库查找结果,并将结果更新到缓存,并返回结果。

每次从数据库获取结果,一般用于 update方法
key 属性要与之前 保持一致

属性

参数

说明

value

缓存的名称,在 spring 配置文件中定义,必须指定至少一个

key

缓存的 key,可以为空,如果指定要按照 SpEL 表达式编写,如果不指定,则缺省按照方法的所有参数进行组合

condition

缓存的条件,可以为空,使用 SpEL 编写,返回 true 或者 false,只有为 true 才进行缓存

示例:

@CachePut(value="selectByPrimaryKey",key="'selectOne-'+#bookId",condition="#bookId>90")

@Override

publicBookselectByPrimaryKey(IntegerbookId) {

Bookbook=bookMapper.selectByPrimaryKey(bookId);

returnbook;

}

@CacheEvict

用来清除用在本方法或者类上的缓存数据(用在哪里清除哪里)。

属性

参数

说明

value

缓存位置的一段名称,不能为空

key

缓存的key,默认为空,表示使用方法的参数类型及参数值作为key,支持SpEL

condition

触发条件,满足条件就加入缓存,默认为空,表示全部都加入缓存,支持SpEL

allEntries

true表示清除value中的全部缓存,默认为false

示例

示例一:清空指定缓存

@CacheEvict(value="selectByPrimaryKey")

@Override

publicvoiddelete(IntegerbookId) {

System.out.println("被删除了...");

}

示例二:清空所有缓存

@CacheEvict(value="selectByPrimaryKey",allEntries=true)

@Override

publicvoiddelete(IntegerbookId) {

System.out.println("被删除了...");

}

多框架->配置文件覆盖问题

引入多框架导致配置文件覆盖

后者覆盖前者

比如,在 spirng.xml 文件引入过个相关文件

spring-mybatis 文件会引入 jdbc.properties 配置

spring-redis 文件会引入 redis.properties 配置

从而导致, redis 会覆盖 jdbc的连接

解决问题

将spring-mybatis.xml和spring-redis.xml中引入的properties配置文件注释,并统一到spring.xml文件中配置引入。

PS: spring-mybatis 的引入语句需要注释,否则 重复引用了
spring-redis 的引入语句也要同步注释。

class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">

classpath:jdbc.properties

classpath:redis.properties

本节回顾

Spring 与 redis 集成配置

  1. 导入依赖

spring-data-redis
jedis
jackson
  1. spring-redis.xml

  1. 导入外部的配置文件 redis.properties

  1. properties 多文件引入方式的改变

  1. 创建数据库连接池

  1. 创建 ConnectionFactory 连接工厂类

  1. 创建 RedisTemplate

  1. 实践操作

@Autowire

privateRedisTemplateredisTemplate;

redisTemplate.opsForValue().set(key,value);

开启注解式缓存

  1. 配置 缓存管理器 RedisCacheManager

  1. 通过 ref 引用 RedisTemplate ref="RedisTemplate"

  1. 设置统一过期时间:3600s

  1. 开启缓存前缀 -cache-

  1. 配置自定义的 KeyCacheGenerator 的生成器

-cache- 类名.方法名:参数:不同方法(相同方法)的不同参数所缓存的效果不一样

  1. 开启注解式缓存

案例

关键点

  • @Cacheable:第一次先到缓存查,缓存有返回,如果缓存没有触发真实查询,将数据存储到缓存里面再返回。(第二次查询只要数据有缓存,且没过期就从缓存返回)

  • @CachePut:每一次都触发数据库真实查询,并将查询结果存储到缓存中。实现数据更新和同步

  • @CacheEvict:

  • 根据指定 Key,删除某个缓存

  • 批量的删除,删除所有

相关知识

key的生成策略

keyGenerator

1)-cache-BookServiceImpl.queryBookPager:0 无参2)-cache-BookServiceImpl.queryBookPager:zs,10 ??啥情况3)-cache-BookServiceImpl.queryBookPager:53 参数为 null值,赋值常量4)-cache-BookServiceImpl.queryBookPager:89,90,91... 判断参数是否为集合或集合,循环遍历取出单个参数5)-cache-BookServiceImpl.queryBookPager:zs String类型,直接拼接参数6)-cache-BookServiceImpl.queryBookPager:109 HashCode值

雪崩、击穿和穿透

缓存雪崩和缓存击穿主要是数据不在缓存上,而缓存穿透是数据既不在缓存上,也不在数据上。

详情参考:https://blog.csdn.net/weixin_39559282/article/details/114167769

手动配置缓存

//进行 redis缓存 RedisTemplate:既可以操控 String,也可操控 Hash

@Autowired

privateRedisTemplateredisTemplate;//Bean 定义的 id=redisTemplate

@Autowired

// private StringRedisTemplate stringRedisTemplate;//只能操控 String

@Test

publicvoidredisTest(){

Bookbook=newBook();

book.setBookId(10);

book.setBookName("三字经");

book.setBookNamePinyin("szj");

book.setBookPrice(120.0F);

book.setBookType("经书");

//这里存储的 书本对象:redis 内不允许直接存储对象,根据 String将 书本对象转成 json后再存进 redis

redisTemplate.opsForValue().set("book",book,15L, TimeUnit.SECONDS);//手动设置 时间:15秒后过期

// redisTemplate.opsForValue().set("book",book);//String 类型

// redisTemplate.opsForHash();//Hash类型

}

相关内容

热门资讯

“日新月异”造句 1、写诗贵在能够灵活,灵活才能日新月异。2、那最神圣恒久而又日新月异的,那最使我们感到惊奇和震撼的两...
“急中生智”造句 1、 突遇危险,他急中生智化险为夷了。2、 大火马上就要烧进来了,艳艳急中生智,披着一条沾湿的被子冲...
一知半解的意思及造句 一知半解的意思及造句  【一知半解的拼音】:  yī zhī bàn jiě  【一知半解的意思】:...
“囟门”造句 1、 结论前囟门皮样囊肿是良性先天性发育异常疾病,诊断明确后手术治疗效果好。2、 小儿囟门未闭合时,...
用兴高采烈造句 用兴高采烈造句精选  1、肯尼科特兴高采烈地说,“今年夏天我们可要玩个痛快。”  2、庆“六一”的游...
日积月累造句   日积月累造句  1、因为在感觉,知觉和抉择上日积月累的超刺激的冲击已经在我们中间酿成了疾病。  ...
豁然开朗造句 豁然开朗造句大全  造句,动词词语,是指用词语组织句子。今亦以指初等学校语文练习内容之一。下面是关于...
机灵造句 机灵造句  在平日的学习、工作和生活里,大家最不陌生的就是句子了吧,根据语气的不同句子可以分为陈述句...
“得过且过”造句 1、人的一生是很短促的,应该珍惜生命,努力工作,不要得过且过。2、随缘不是得过且过,因循苟且,而是尽...
收成的造句 收成的造句  在日常的学习中,是不是经常追着老师要知识点?知识点是知识中的最小单位,最具体的内容,有...
“杂乱无章”造句 101、明末清初这段历史杂乱无章,读起来千头万序,让人摸不着头脑。102、图一无论在枝干的构成上,还...
用寒暄造句 用寒暄造句  造句是初等学校语文练习内容、考试题型、作业方式等之一。以下是小编为大家整理的用寒暄造句...
裤腰的意思及造句 裤腰的意思及造句  裤腰拼音  【注音】: ku yao  裤腰解释  【意思】:裤子的最上端,系腰...
绿油油如何造句 绿油油如何造句  1.百合花的叶子十分漂亮,绿油油的,衬托着花朵。绿叶不可说四季常青,但绿起来却绿着...
三年级与众不同意思及造句   桂林的水独一无二,桂林的山与众不同。下面是小编为你带来的三年级与众不同意思及造句,欢迎阅读。  ...
暴厉恣睢的反义词 暴厉恣睢的反义词有:慈眉善目,暴厉恣睢[bào lì zí suī]的意思:暴:残暴;恣睢:横行霸道...
入超拼音解释及造句 入超拼音解释及造句  入超拼音  【注音】: ru chao  入超解释  【意思】:在一定时期(一...
今非昔比的反义词 今非昔比的反义词有:今不如昔,厚古薄今,慕古薄今,今非昔比[jīn fēi xī bǐ]的意思:昔:...
独具只眼的反义词 独具只眼的反义词有:一无所长,愚昧无知,独具只眼[dú jù zhī yǎn]的意思:具有独到的见解...
不以为然的反义词 不以为然的反义词有:五体投地,仰承鼻息,理所当然,顶礼膜拜,不以为然[bù yǐ wéi rán]的...