Spring Boot整合Redis笔记

news2024/12/28 19:26:30

在这里插入图片描述

文章目录

  • 前言
  • Java 操作 Redis
    • Jedis 操作-测试
    • Jedis 实例-手机验证码
  • Redis与Spring Boot整合
    • 整合步骤
  • Redis 的事务操作
    • Redis的事务定义
    • Multi、Exec、discard 基本命令
    • 事务冲突的问题
      • 为什么要做成事务
      • 悲观锁
      • 乐观锁
        • WATCH key [key ... ]
    • Redis事务三特性
    • Redis事务秒杀案例
      • 解决计数器和人员记录的事务操作

前言

📋前言📋
💝博客:【无聊大侠hello word】💝
✍有一点思考,有一点想法,有一点理性!✍
✍本文由在下【无聊大侠hello word】原创,首发于CSDN✍

Java 操作 Redis

Jedis 操作-测试

  1. 引入依赖
<!--jedis操作-->
<dependency>
   <groupId>redis.clients</groupId>
   <artifactId>jedis</artifactId>
   <version>3.8.0</version>
</dependency>

<!--Test依赖-->
<dependency>
   <groupId>junit</groupId>
   <artifactId>junit</artifactId>
   <version>4.13.2</version>
   <scope>compile</scope>
</dependency>
  1. 简单测试
package com.jedis.example;

import org.junit.Test;
import redis.clients.jedis.Jedis;

import java.util.List;
import java.util.Set;

public class JedisDemo1
{

    public static void main(String[] args)
    {
        // 创建Jedis对象
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 测试
        String value = jedis.ping();
        System.out.println(value);
        
		jedis.close();
    }

    /**
     * 描述:操作 key [String]
     */
    @Test
    public void Demo1()
    {
        // 创建Jedis对象
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 添加
        jedis.set("name", "lucy");

        // 获取
        String name = jedis.get("name");
        System.out.println(name);

        // 设置多个key-value
        jedis.mset("k1", "v1", "k2", "v2");
        List<String> mget = jedis.mget("k1", "k2");
        System.out.println(mget);


        Set<String> keys = jedis.keys("*");
        for (String key : keys)
        {
            System.out.println(key);
        }
        
		jedis.close();
    }

    /**
     * 描述:操作 List [List]
     */
    @Test
    public void Demo2()
    {
        // 创建Jedis对象
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 添加List数据
        jedis.lpush("key1", "lucy", "mary", "jack");
        List<String> list = jedis.lrange("key1", 0, -1);
        System.out.println(list);
                
		jedis.close();
    }

    /**
     * 描述:操作 set
     */
    @Test
    public void Demo3()
    {
        // 创建Jedis对象
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 添加Set数据
        jedis.sadd("names", "lucy");
        jedis.sadd("names", "jack");

        Set<String> name = jedis.smembers("name");
        System.out.println(name);
                
		jedis.close();
    }

    /**
     * 描述:操作 Hash
     */
    @Test
    public void Demo4()
    {
        // 创建Jedis对象
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 添加Hash数据
        jedis.hset("users", "age", "20");

        String value = jedis.hget("users", "age");
        System.out.println(value);
                
		jedis.close();
    }

    /**
     * 描述:操作 zset
     */
    @Test
    public void Demo5()
    {
        // 创建Jedis对象
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 添加zset数据
        jedis.zadd("china", 100d, "shanghai");

        Set<String> value = jedis.zrange("china", 0, -1);
        System.out.println(value);
        
		jedis.close();
    }
}

更多Redis数据类型命令操作

Jedis 实例-手机验证码

要求:

1、输入手机号,点击发送后随机生成6位数字码,2分钟有效

2、输入验证码,点击验证,返回成功或失败

3、每个手机号每天只能输入3次

只实现Java操作

在这里插入图片描述
分析:
在这里插入图片描述

代码实现:

package com.jedis.example;

import redis.clients.jedis.Jedis;

import java.util.Random;

/**
 * 描述:Jedis 实例-手机验证码
 *
 * @author 为一道彩虹
 */
public class PhoneCode
{
    /**
     * 描述:1.生成6位数字验证码
     */
    public static String getCode()
    {
        Random random = new Random();
        String code = "";
        for (int i = 0; i < 6; i++)
        {
            int rand = random.nextInt(10);
            code += rand;
        }
        return code;
    }

    /**
     * 描述:2.每个手机每天只能发送三次,验证码放到redis中,设置过期时间120秒
     */
    public static void verifyCode(String phone, String code)
    {
        // 连接redis
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 拼接key [制定规则]
        // 手机号发送次数key
        String countKey = "VerifyCode" + phone + ":count";

        // 验证码key
        String codeKey = "VerifyCode" + phone + ":code";

        // 每个手机每天只能发送三次
        String count = jedis.get(countKey);
        if (count == null)
        {
            // 没有发送次数,第一次发送, 设置发送次数是1
            jedis.setex(countKey, 24*60*60, "1");
        }
        else if(Integer.parseInt(count) <= 2)
        {
            // 发送次数+1
            jedis.incr(countKey);
        }
        else if (Integer.parseInt(count) > 2)
        {
            // 发送三次,不能再发送
            System.out.println("今天发送次数已经超过三次");
            jedis.close();
            return;
        }

        // 发送验证码放到redis里面
        jedis.setex(codeKey, 120, code);
        jedis.close();
    }

    /**
     * 描述:3.验证码校验
     */
    public static void getRedisCode(String phone, String code)
    {
        // 连接redis
        Jedis jedis = new Jedis("127.0.0.1", 6379);

        // 验证码key
        String codeKey = "VerifyCode" + phone + ":code";

        // 从redis获取验证码
        String redisCode = jedis.get(codeKey);

        // 判断
        if (redisCode.equals(code))
        {
            System.out.println("成功");
        }
        else{
            System.out.println("失败");
        }

        jedis.close();
    }

    public static void main(String[] args)
    {
        // 生成6位数字验证码
        String code = getCode();

        // 模拟验证码发送
         verifyCode("1613075408", code);

        // 验证码校验
//        getRedisCode("1613075408", "466604");
    }
}

在这里插入图片描述

Redis与Spring Boot整合

Spring Boot整合Redis非常简单,只需要按如下步骤整合即可

整合步骤

1、在pom.xml文件中引入redis相关依赖

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

<!--spring2.X集成redis 所需common-pooL2-->
<dependency>
   <groupId>org.apache.commons</groupId>
   <artifactId>commons-pool2</artifactId>
   <version>2.6.0</version>
</dependency>

2、application.properties配置redis配置

#Redis服务器地址
spring.redis.host=127.0.0.1
#Redis服务器连接端口
spring.redis.port=6379
#Redis数据库素引(默认为0)
spring.redis.database=0
#连接超时时间(毫秒)
spring.redis.timeout=1800000
#连接池最太连接数(使用负值表示没有限制)
spring.redis.lettuce.pool.max-active=20
#最大阻塞等待时间(负数表示没限制)
spring.redis.lettuce.pool.max-wait=-1
#连接池中的最大空闲连接
spring.redis.lettuce.pool.max-idle=5
#连接池中的最小空闲连接
spring.redis.lettuce.pool.min-idle=0

3、添加redis配置类

package com.redis.spring.boot.config;

import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.cache.CacheManager;
import org.springframework.cache.annotation.CachingConfigurerSupport;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheConfiguration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.RedisSerializationContext;
import org.springframework.data.redis.serializer.RedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;

import java.time.Duration;

/**
 * 描述:Redis配置类
 *
 * @author 为一道彩虹
 */
@EnableCaching
@Configuration
public class RedisConfig extends CachingConfigurerSupport
{
    @Bean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory)
    {
        RedisTemplate<String, Object> template = new RedisTemplate<>();
        RedisSerializer<String> redisSerializer = new StringRedisSerializer();
        Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
        ObjectMapper om = new ObjectMapper();
        om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
        om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        jackson2JsonRedisSerializer.setObjectMapper(om);

        template.setConnectionFactory(factory);
        // key序列化方式
        template.setKeySerializer(redisSerializer);
        // value序列化
        template.setValueSerializer(jackson2JsonRedisSerializer);
        // value hashmap序列化
        template.setHashValueSerializer(jackson2JsonRedisSerializer);

        return template;
    }

    @Bean
    public CacheManager cacheManager(RedisConnectionFactory factory)
    {
        RedisSerializer<String> redisSerializer = new StringRedisSerializer();
        Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);

        // 解决查询缓存转换异常的问题
        ObjectMapper om = new ObjectMapper();
        om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
        om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        jackson2JsonRedisSerializer.setObjectMapper(om);

        // 配置序列化(解决乱码的问题),过期时间600秒
        RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig()
                .entryTtl(Duration.ofSeconds(600))
                .serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(redisSerializer))
                .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer))
                .disableCachingNullValues();

        RedisCacheManager cacheManager = RedisCacheManager.builder(factory)
                .cacheDefaults(config)
                .build();

        return cacheManager;
    }

}

4、测试一下

package com.redis.spring.boot.controller;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

/**
 * 描述:RedisTestController
 *
 * @author 为一道彩虹
 */
@RestController
@RequestMapping("/redisTest")
public class RedisTestController
{
    @Autowired
    private RedisTemplate redisTemplate = null;

    @GetMapping
    public String testRedis()
    {
        // 设置值到Redis
        redisTemplate.opsForValue().set("name", "lucy");

        // 从Redis获取值
        Object name = redisTemplate.opsForValue().get("name");
        return name.toString();
    }
}

在这里插入图片描述

Redis 的事务操作

Redis的事务定义

Redis 事务是一个单独的隔离操作:事务中的所有命令都会序列化、按顺序地执行。事务在执行的过程中,不会被其他客户端发送来的命令请求所打断。

Redis 事务的主要作用就是串联多个命令防止别的命令插队。

Multi、Exec、discard 基本命令

从输入Multi 命令开始,输入的命令都会依次进入命令队列中,但不会执行,直到输入Exec后,Redis会将之前的命令队列中的命令依次执行。

组队的过程中可以通过discard来放弃组队。

在这里插入图片描述

事务的错误处理

  1. 组队中某个命令出现了报告错误,执行时整个的所有队列都会被取消。

在这里插入图片描述

  1. 如果执行价段果个命令报出了错误,则只有报错的命令不会被执行,而其也的命令都会执行,不会回滚。

在这里插入图片描述

事务冲突的问题

为什么要做成事务

想想一个场景:有很多人有你的账户,同时去参加双十一抢购

例子:

  • 一个请求想给金额减8000
  • 一个请求想给金额减5000
  • 一 个请求想给金额减1000

在这里插入图片描述

悲观锁

在这里插入图片描述

悲观锁( Pessimistic Lock),顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会 block 直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁表锁等,读锁写锁等,都是在做操作之前先上

乐观锁

在这里插入图片描述

乐观锁 (Optimistic Lock),顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量。Redis就是利用这种check-and-set机制实现事务的

WATCH key [key … ]

在执行 multi 之前,先执行 watch key1 [key2],可以监视一个(或多个)key,如果在事务执行之前这个(或这些)key被其他命令所改动,那么事务将被打断。

Redis事务三特性

单独的隔离操作:

  • 事务中的所有命令都会序列化、按顺序地执行。事务在执行的过程中,不会被其他客户端发送来的命令请求所打断。

没有隔离级别的概念:

  • 队列中的命令没有提交之前都不会实际被执行,因为事务提交前任何指令都不会被实际执行

不保证原子性:

  • 事务中如果有一条命令执行失败,其后的命令仍然会被执行,没有回滚

Redis事务秒杀案例

解决计数器和人员记录的事务操作

在这里插入图片描述

先赞后看,养成习惯!!!^ _ ^ ❤️ ❤️ ❤️
码字不易,大家的支持就是我的坚持下去的动力。点赞后不要忘了关注我哦!

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

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

相关文章

分布式定时任务-XXL-JOB-教程+实战

一.定时任务概述 1.定时任务认识 1.1.什么是定时任务 定时任务是按照指定时间周期运行任务。使用场景为在某个固定时间点执行&#xff0c;或者周期性的去执行某个任务&#xff0c;比如&#xff1a;每天晚上24点做数据汇总&#xff0c;定时发送短信等。 1.2.常见定时任务方案…

docker-compose容器编排部署

docker-compose部署微服务1、Docker-Compose是什么&#xff1f;2、应用场景3、docker-compose部署SpringBoot项目3.1 编写Dockfile3.2 编写docker-compose.yaml3.3 修改工程配置3.4 将相关文件上传到服务器3.5 执行docker-compose up本文是对DockerNginx打包部署前后端分离项目…

E5061B矢量网络分析仪VNA概念

矢量网络分析仪VNA是一种测试仪器&#xff0c;它可以将网络的响应测量成矢量:实参数和虚参数&#xff0c;从而表征其性能。矢量网络分析仪VNA是射频设计实验室和许多制造和服务领域的重要测试仪器。虽然矢量网络分析仪主要侧重于研究和开发&#xff0c;但它也可以为所有类型的R…

2月3日 读书笔记

我们将程序改善一下&#xff0c;让程序在按下一个键后不结束&#xff0c;而是把所按键的编码在画面上显示出来&#xff0c;这样就可以切实完成中断处理程序了。 所谓中断处理&#xff0c;基本上就是打断CPU本来的工作&#xff0c;加塞要求进行处理。而且处理中断期间不再接收别…

创业30载,百亿市值奥瑞金未来可期

1994年&#xff0c;关玉香在海南文昌破土兴建海南奥瑞金包装实业有限公司&#xff08;原名&#xff1a;文昌奥瑞金制罐公司&#xff09;&#xff0c;与儿子周云杰一起带领着16名工人进入了金属包装行业&#xff0c;从0到如今的百亿市值&#xff0c;就此拉开了一路“封神”的序幕…

字符串(一)BF算法与KMP算法

给一个主串s&#xff0c;在给一个子串substr&#xff0c;判断substr是否为s的子串 一、BF 暴力搜索 暴力&#xff0c;依次逐个比较字符&#xff0c;先从主串和模式串的第一个字符开始&#xff0c;如果相等一起比较下一个字符&#xff0c;如果不相等&#xff0c;那么重新回到模…

PTA L1-032 Left-pad(详解)

前言&#xff1a;本期是关于L1-032 Left-pad的详解&#xff0c;内容包括四大模块&#xff1a;题目&#xff0c;代码实现&#xff0c;大致思路&#xff0c;代码解读&#xff0c;今天你c了吗&#xff1f; 题目&#xff1a; 根据新浪微博上的消息&#xff0c;有一位开发者不满NPM…

SpringCloud Alibaba—— 微服务网关GateWay

目录 1、GateWay网关概述 1.1、什么是GateWay&#xff1f; 1.2、为什么要使用微服务网关&#xff1f; 1.3、Zuul与GateWay网关的区别&#xff1f; 2、快速入门 2.1、创建项目 2.2、配置yml文件 2.3、controller层 2.4、启动类 2.5、启动整体项目 2.6、配置全局过滤器…

代码随想录算法训练营第十七天 | 110.平衡二叉树,257. 二叉树的所有路径,404.左叶子之和

一、昨日回顾与补充今天看了Day16讲解的视频&#xff0c;对于求二叉树最大深度、最小深度以及求完全二叉树的节点个数有了新的理解&#xff0c;总结如下&#xff1a;1.深度和高度的区别&#xff08;之前就看看定义忽略了&#xff09;二叉树节点的深度&#xff1a;指从根节点到该…

jvm垃圾收集器有哪些

Serial收集器 Serial收集器是最基本&#xff0c;发展最悠久的收集器&#xff0c;在JDK1.3.1之前是虚拟机新生代垃圾回收的唯一选择。这个收集器是一个单线程的。它的单线程的意义并不仅仅说明它只会使用一个CPU或者一条收集线程去完成收集工作&#xff0c;最重要的是&#xff…

量化选股股票和量化交易的两者有什么区别?

最近a股在频繁波动后终于开始走强。近几个月来&#xff0c;一些净值明显下降的量化产品业绩终于实现了阶段性回升。与此同时&#xff0c;根据金融终端&#xff0c;量化私募管理的规模已经超过1万亿元。面对目前市场上大量超额收益和大幅波动带来的不合理定价带来的收益&#xf…

磨金石教育摄影技能干货分享|春节街拍欣赏:大街上的年味

当然还有很多人依然沉浸在过年的喜庆氛围中&#xff0c;我们一般把除夕到十五这个期间段都归入春节期间。在这个时间段内街道上都是喜庆的氛围热热闹闹&#xff0c;年味十足。过了除夕之后&#xff0c;春节娱乐的主题除了走亲访友&#xff0c;就是出行了。春节小长假&#xff0…

jmeter脚本处理加密验签

一 、账号密码加密 1.1 背景&#xff1a; 一个登录接口&#xff0c;账号是明文传输&#xff0c;但是密码要先经过加密&#xff0c;再做传输。 比如&#xff1a; 一个用户&#xff0c;账号为123465 密码为 abcde 实际上登录接口&#xff0c;请求&#xff0c;传参为 账号 1…

使用傅里叶级数展开法从谐波的和中产生方波(Matlab代码实现)

目录 &#x1f4a5;1 概述 &#x1f4da;2 运行结果 &#x1f389;3 参考文献 &#x1f468;‍&#x1f4bb;4 Matlab代码 &#x1f4a5;1 概述 谐波是指对周期性非正弦交流量进行傅里叶级数分解所得到的大于基波频率整数倍的各次分量&#xff0c;通常称为高次谐波&#xf…

C++11 可变参数模板

作者&#xff1a;小萌新 专栏&#xff1a;C进阶 作者简介&#xff1a;大二学生 希望能和大家一起进步&#xff01; 本篇博客简介&#xff1a;介绍C11的可变参数模板 可变参数模板可变参数模板的概念可变参数模板的定义方式参数包两种解开方式递归展开参数包逗号表达式展开参数包…

Springboot在线考试管理系统

一、项目简介 本项目是一套基于Springboot在线考试管理系统&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Java学习者。 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;…

java基于springboot+vue的校园跑腿系统

随着我国教育制度的改革的发展&#xff0c;各大高校的学生数量也在不断的增加。当前大学生的生活和消费习惯等导致他们更喜欢通过网络来获取自己想要的商品和服务&#xff0c;这也是导致当前校园跑腿盛行的主要原因。为了能够让更多的学生享受到校园跑腿的服务&#xff0c;我们…

中睿天下荣获中国网络安全产业联盟(CCIA)2022年度先进会员单位奖

近日&#xff0c;中国网络安全产业联盟&#xff08;CCIA&#xff09;发布“关于2022年度表彰先进的决定”&#xff0c;对在联盟工作中做出积极贡献的会员单位进行表彰&#xff0c;中睿天下荣获中国网络安全产业联盟“2022年度先进会员单位“奖。中国网络安全产业联盟成立于2015…

Linux基本使用

文章目录1. Linux 基本命令文件内容查看命令wget下载tar.gz文件tar压缩解压缩yum2. 安装node/npm下载解压配置环境变量设置npm代理3. 权限1. Linux 基本命令 ls&#xff1a;用来查看当前目录文件 # 查看隐藏文件 -a ls -a # 查看详细信息 -l 简写为 ll ls -l ll # 查看指定目…

Docker - 8. Docker镜像底层原理

目录 1. 镜像是什么 2. UnionFS 联合文件系统 3. Docker镜像加载原理 4. Docker 容器和镜像的底层关系 5. Docker 镜像 commit 操作案例 1. 镜像是什么 Docker 镜像是一个特殊的文件系统&#xff0c;除了提供容器运行时所需的程序、库、资源、配置等文件外&#xff0c;还…