redis序列化_SpringBoot整合redis

v2-5cbc76410fb5ff72a4e5da87d86495be_1440w.jpg?source=172ae18b

redis是最常用的缓存数据库,常用于存储用户登录token、临时数据、定时相关数据等。

redis是单线程的,所以redis的操作是原子性的,这样可以保证不会出现并发问题。

redis基于内存,速度非常快,据测试,redis读的速度是110000次/s,写的速度是81000次/s

本节介绍SpringBoot引入redis,以及使用RedisTemplate来操作redis数据。

采用SpringBoot 2.1.9.RELEASE,对应示例代码在:https://github.com/laolunsi/spring-boot-examples

一、A Simple Demo-使用SpringBoot连接redis

maven:

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

yml:

server:port: 8867
spring:redis:host: localhostport: 6379#password: ''database: 6

测试类:

@SpringBootTest
@RunWith(SpringRunner.class)
public class RedisTest {@Autowiredprivate RedisTemplate redisTemplate;@Testpublic void testRedis() {String key = "hello";redisTemplate.opsForValue().set("hello", "你好");String res = (String) redisTemplate.opsForValue().get(key);System.out.println(res);}
}

执行结果:

v2-652e8eb475b25e91e004b1588e94b409_b.jpg

看一下redis:

v2-6a38390e87cf413e45d42704eda65c21_b.jpg

这里存在一个问题:默认的存储方式导致key在redis-manager里面显示出来是乱码的,并且存储结果是二进制了。这样不利用我们查看redis里面的数据。

我们需要自定义redis存储的序列化规则。

二、解决RedisTemplate默认序列化的问题

完善一下maven:

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter</artifactId>
</dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope>
</dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId>
</dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

定义RedisConfig类:

/*** redis配置* 主要是配置Redis的序列化规则,替换默认的jdkSerializer* key的序列化规则用StringRedisSerializer* value的序列化规则用Jackson2JsonRedisSerializer*/
@Configuration
public class RedisConfig {@Beanpublic RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory connectionFactory) {RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();redisTemplate.setConnectionFactory(connectionFactory);// 使用Jackson2JsonRedisSerialize替换默认序列化Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);ObjectMapper objectMapper = new ObjectMapper();objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);jackson2JsonRedisSerializer.setObjectMapper(objectMapper);// 设置key和value的序列化规则redisTemplate.setKeySerializer(new StringRedisSerializer());redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);redisTemplate.setHashKeySerializer(new StringRedisSerializer());redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);redisTemplate.afterPropertiesSet();return redisTemplate;}
}

删除之前的key,重新执行一下test方法:

v2-a1c996295819910d76a327d07113ccb8_b.jpg

下面来演示一下SpringBoot使用RedisTemplate进行redis数据的操作

三、基于SpringBoot的redis操作——key/list/hash

RedisTemplate内置redis操作如下:

v2-cd848adac5ea7f827bc51b924d77a8ff_b.jpg

这里主要展示value/hash/list三种用法:

3.1 RedisTemplate.opsForValue

  @Testpublic void testKeyOps() {// 测试redis操作key-value形式Set<String> keySet = new HashSet<>();String key1 = "name";keySet.add(key1);// 存储简单的key-value,并设置过期时间redisTemplate.opsForValue().set(key1, "eknown", 1, TimeUnit.MINUTES);String key2 = "token:user1";String key3 = "token:user2";keySet.add(key2);keySet.add(key3);//redisTemplate.opsForValue().set(key2, "{"name":"eknown"}, "role":"admin"");redisTemplate.opsForValue().set(key3, "{"name":"test"}, "role":"test"");// 根据key的集合获取多个valueList<String> valueList = redisTemplate.opsForValue().multiGet(keySet);for (String value : valueList) {System.out.println(value);}}

执行结果:

v2-37c17e82176f52cb7fa062a47e455103_b.jpg

redis中的数据:

v2-ea3737752837e2ce6cd9ce164c85980f_b.jpg

redis中的key显示出了一个层级关系,这个小技巧对于实际项目有个非常好的作用:通过prefix:suffix这样的形式,可以将redis中存储的数据分出层级。

3.2 RedisTemplate.opsForHash

清空该database下的数据,测试redisTemplate.opsForHash:

    @Testpublic void testHashOps() {String key = "hash";// 单次往hash中存放一个数据redisTemplate.opsForHash().put(key, "1", "你好");Map<String, Object> map = new HashMap<>();map.put("2", "hello");map.put("3a", "china1=2");// 一次性向hash中存放一个mapredisTemplate.opsForHash().putAll(key, map);// 获取hash下的所有key和valueMap<String, Object> resultMap = redisTemplate.opsForHash().entries(key);for (String hashKey : resultMap.keySet()) {System.out.println(hashKey + ": " + resultMap.get(hashKey));}}

执行结果:

v2-5a68b4be724ad33ec42a7c04bddfb82c_b.jpg

redis:

v2-3df024726ca027be7b5cee03befab6fa_b.jpg

3.3 RedisTemplate.opsForList

    @Testpublic void testListOps() {String listKey = "list";redisTemplate.opsForList().leftPush(listKey, "first value"); // 从list最左边插入数据redisTemplate.opsForList().leftPush(listKey, "second value but left");redisTemplate.opsForList().rightPush(listKey, 3); // 从list最右边插入数据List<Object> list = new ArrayList<>();list.add("hello");list.add("http://www.eknown.cn");list.add(23344);list.add(false);redisTemplate.opsForList().rightPushAll(listKey, list); // 从list右边批量插入数据long size = redisTemplate.opsForList().size(listKey);if (size > 0) {for (int i = 0; i < size -1 ; i++) {// 从list最左边开始读取list中的数据,注意pop会导致出栈,也就是数据被取出来了(redis中就没有这个值了)// 此处我们读取size-1条数据,仅留下最后一条数据System.out.println(i + ":" + redisTemplate.opsForList().leftPop(listKey).toString());}}}

执行上面的脚本,注意在最后的读取list数据代码前面加一个断点,此时redis中是这样的:

v2-3ee1b0d7560877e292fd77046325dbad_b.jpg

放开断点,程序继续执行,控制台如下:

v2-c1aa0c293e1a60226f7734d1101c21d3_b.jpg

注意,此时redis中仅剩余最后一条数据,这是由于pop的问题,list中的数据被读取并删除了:

v2-7d7f29861a6a8fbb03a9ee04d2d2f940_b.jpg

好了,这一节主要讲了SpringBoot引入redis,以及使用redis的一些基本操作和相关技巧,在此基础上,我们可以让我们的项目变得更加快速、灵活!


交流学习

我的个人网站:http://www.eknown.cn

Git仓库地址:https://github.com/laolunsi

另外也欢迎大家关注我的公众号:猿生物语,一起学习Java/SpringBoot/SpringCloud技术。

v2-835fe3628eb78c4164658029839b8927_b.jpg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/504360.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

spring cloud alibaba_SpringCloudAlibaba与Cloud搭配方案

一 简介Spring Cloud Alibaba致力于提供微服务开发一站式解决方案。此项目包括开发分布式应用微服务的必需组件&#xff0c;方便开发者通过 Spring Cloud 编程模型轻松使用这些组件来开发分布式应用服务。依托 Spring Cloud Alibaba&#xff0c;你只需要添加一些注解和少量配置…

php curl header_PHP中的yield与协程(二十一节)

大家好&#xff0c;我是老李。顺风说骚话&#xff0c;逆风讲道理最近在大家一起努力下&#xff0c;那个沙雕肺炎患病人数增长率下降了不少&#xff0c;总体来说还算顺&#xff0c;所以今天这篇注定又要骚话连篇了。听说最近不少玉米开始向大连、威海、烟台方向涌入&#xff0c;…

用稳压管保护单片机引脚_一步一步,全程揭开单片机的原理,让做电子变得轻松自如!...

学习单片机必要的硬件学习单片机都需要什么&#xff1f;首先要掌握必要的电子基础知识&#xff1b;再次选择大众化、性价比高的单片机&#xff0c;熟悉它的引脚功能定义&#xff1b;另外还需要电脑与下载程序的下载器。一、单片机建议选择型号为STC89C52RC单片机&#xff0c;如…

python word 表格宽度_RPA手把手——python-docx 设置 word 文档中表格格式

艺赛旗|做RPA生态先行者 RPA10.0全新首发免费下载 点击下载 引入会用到的库 from docx import Document from docx.shared import Pt from docx.enum.text import WD_PARAGRAPH_ALIGNMENT from docx.shared import Cm from docx.shared import RGBColor 表格样式 #方法一&#…

软引用和弱引用的区别_强、软、弱、虚引用的区别和使用

原文阅读&#xff1a;强、软、弱、虚引用的区别和使用​mp.weixin.qq.comJava提供了四种级别的应用类型&#xff1a;强引用、软引用、弱引用及虚引用。那么这四种引用类型有什么区别呢&#xff1f;首先我们通过一张图来看看四种引用在Java中的表示&#xff1a;FinalReference由…

python文本文件不能用二进制文件方式读入_如何使用python函数以二进制形式读取文件?...

虽然读取文件的方式各种各样&#xff0c;但是通过二进制&#xff0c;还是头一次&#xff0c;实现过过程并不难&#xff0c;我们需要将文件先做好封存&#xff0c;以函数的形式&#xff0c;保存下来&#xff0c;然后直接导入进行使用&#xff0c;这样就可以读取文件&#xff0c;…

一组数字中算出最相近的组合_据说在金字塔里发现的这组数字,貌似是通往宇宙的密码...

我们都知道阿拉伯数字是全世界都在用的计数单位的数字&#xff0c;我们的生活离不开阿拉伯数字&#xff0c;其广泛性很大。比如买菜、买衣服。做数学题等等都需要用到。不过呢早前有人称在埃及金字塔内发现了一组数字142857&#xff0c;这组数字貌似就是通往宇宙的密码。这一说…

前端悬浮窗效果_Flutter自绘组件:微信悬浮窗(一)

看微信公众号的时候时常会想退出去回复消息&#xff0c;但又不想放弃已经阅读一半的文章&#xff0c;因为回复信息后再从公众号找到该篇文章之间有不必要的时间花费&#xff0c;微信悬浮窗的出现解决了这个烦恼&#xff0c;回复完消息之后只需要点击悬浮窗就可以回到之前在阅读…

date类型_Quartz与Date---cron的相互转换

产生原因:因为项目最近设计到了一个Quartz相关的模块&#xff0c;前端需要传递时间参数到后台, 然后后台设置一个新的定时任务, 所以后台需要一个可以实现Date与cron之间的相互转换(因为Quartz需要的Cron格式的数据)&#xff0c;所以就借助java的SimpleDateFormat的格式化,然后…

vbs if 不等于_6、if语句和关系表达式

示例3.1&#xff1a;星星公司致力于信件快递业务&#xff0c;收费标准是&#xff1a;500g以内6元&#xff0c;超过500g9元。应该就是输入重量&#xff0c;显示钱&#xff0c;那可以用cout和cin&#xff0c;如果w小于500&#xff0c;c是6&#xff0c;否则&#xff0c;c是9。//pr…

python文件是怎么写_python头文件怎么写

本文主要以python2为例。首先介绍一下Python头文件的编程风格&#xff0c;然后再给大家详细介绍import部分的基本用法。这两个部分就是Python中头文件的组成模块。编程风格#!/usr/bin/env python #在文件头部 ( 第一行 ) 加上 设置 Python 解释器 # -*- coding: utf-8 -*- #在文…

【学习笔记】第二章——处理机调度的概念、层次、时机、切换过程 调度方式、调度算法的指标

文章目录一. 概念 & 层次1. 高级调度&#xff08;作业调度&#xff09;2. 中级调度&#xff08;内存调度&#xff09;挂起态 & 七状态模型3. 低级调度&#xff08;进程调度&#xff09;4. 三种调度的对比联系 && 总结二. 时机、切换过程 & 调度方式1. 进程…

formdata上传文件_关于multipart/formdata上传文件

最近在做一个文件上传的开放接口&#xff0c;用到Content-Type: multipart/form-data这种请求类型&#xff0c;特地做了一些研究和记录。在最初的 http协议中&#xff0c;并没有上传文件方面的功能。RFC1867为 http协议添加了这个能力。常见的浏览器&#xff0c;如 Microsoft I…

【学习笔记】第二章——调度算法:先来先服务FCFS、短作业优先SJF、高响应比HRRN

文章目录一. 先来先服务&#xff08;FCFS&#xff09;二. 短作业优先&#xff08;SJF&#xff09;三. 高响应比优先1. 对前面两种算法的思考2. 描述四. 一、二、三总结例子都要手动写一遍哦&#xff5e;这三个是供早期的批处理系统使用的算法 一. 先来先服务&#xff08;FCFS&a…

【学习笔记】第二章——时间片轮转RR、优先级调度、多级反馈队列调度算法

文章目录一. 时间片轮转二. 优先级调度三. 多级反馈队列调度算法四. 总结一. 时间片轮转 公平&#xff0c;轮流给进程提供时间片只用于进程调度&#xff08;只有进程才能被分配时间片&#xff09;抢占式&#xff0c;由时钟装置发出时钟中断来通知**缺点&#xff1a;**高频的进…

laravel 分词搜索匹配度_elasticsearch基础笔记9-elasticsearch 词项全文搜索

es的核心功能就是搜索和分析。那么我们看看搜索相关内容1、搜索机制在进入搜索之前&#xff0c;会对查询体根据情况进行分析和处理。2、有哪些常用搜索类型全文查询 词项查询 复合查询 嵌套查询 位置查询 特殊查询等。我们常用到的就是前三种&#xff0c;学起来简单&#xff0c…

【学习笔记】第二章——进程同步、进程互斥、进程互斥的硬件/软件实现方法

文章目录一. 进程同步 && 进程互斥四个区域四个原则总结&#xff1a;二. 进程互斥的软件实现方法1&#xff09;单标志法2&#xff09;双标志先检查法3&#xff09;双标志后检查法4&#xff09;Peterson 算法总结三. 进程互斥的硬件实现方法1&#xff09;中断屏蔽方法2&…

python的数据结构包括那些_python算法与数据结构-什么是数据结构

一、什么是数据结构 数据结构&#xff0c;直白地理解&#xff0c;就是研究数据的存储方式。 我们知道&#xff0c;数据存储只有一个目的&#xff0c;即为了方便后期对数据的再利用&#xff0c;就如同我们使用数组存储 {1,2,3,4,5} 是为了后期取得它们的值&#xff0c;因此&…

seo自动发外链_一套节约成本全网营销方案-小小课堂SEO培训教程

很多公司都是由一些小型工作室或个人工作室慢慢发展而来的&#xff0c;在这过程中&#xff0c;人员、资金链、资源、项目等因素一旦出现问题&#xff0c;可能会导致项目失败&#xff0c;那么在互联网上投入的资金如果过多&#xff0c;可能都是白白打了水漂。今天&#xff0c;小…

python 支付宝个人账单_金融支付财务融合业务-实践分享1:订单、账单、交易流水、账套知识解构、原理解析...

本文作者从实际工作实践出发&#xff0c;结合案例等分享了电商金融支付财务融合中的基本概念和相关原理解析&#xff0c;包括&#xff1a;订单、账单、交易流水和账知识解构&#xff0c;供大家一同参考和学习。从事电商、进销存、金融、支付、财务的产品同学&#xff0c;是否对…