Redis | 基于 Redis 实现机器列表 Token 缓存的 Java 实现
关注:CodingTechWork
引言
在分布式系统中,Token 缓存是一种常见的需求。它可以帮助我们快速验证用户身份,减少对数据库的频繁访问,提高系统的性能和响应速度。本文将介绍如何使用 Redis 来实现机器列表的 Token 缓存,在 Kubernetes Pod 部署的环境中,为了避免多个 Pod 同时执行相同的定时任务(如刷新缓存 Token),我们需要引入分布式锁机制。以下是基于RedisTemplate和分布式锁实现的分布式刷新缓存 Token 的完整 Java 示例代码。
为什么选择 Redis
Redis 是一个高性能的键值存储数据库,它提供了丰富的数据结构和极高的读写速度。以下是选择 Redis 实现 Token 缓存的原因:
- 高性能:Redis 的读写速度非常快,能够轻松应对高并发场景下的 Token 验证请求。
- 持久化支持:虽然 Redis 是内存数据库,但它支持多种持久化方式,可以保证数据在机器故障时不会丢失。
- 易于使用:Redis 提供了丰富的客户端库,方便在各种编程语言中使用。
设计思路
- Token 的生成与存储:使用 UUID 生成唯一的 Token,并通过 RedisTemplate 存储到 Redis 中,同时设置
过期时间。 - Token 的验证:通过 RedisTemplate 从 Redis 中获取 Token,并检查其是否存在和是否过期。
- 分布式锁:使用 Redis 实现分布式锁,确保在分布式环境中只有一个 Pod 能够执行 Token 刷新任务。
- 定时刷新 Token:使用 Spring 的
@Scheduled注解实现定时任务,结合分布式锁确保同一时间只有一个 Pod 执行 Token 刷新操作。(当然,我们也可以引入xxl-job组件来实现定时任务)
Java 实现
引入依赖
在 Maven 项目中,需要引入 Spring Boot Starter Data Redis`` 和 Spring Boot Starter Scheduling 的依赖。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-scheduling</artifactId>
</dependency>
Redis 配置类
创建一个 Redis 配置类,用于配置 RedisTemplate。
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;@Configuration
public class RedisConfig {@Beanpublic RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {RedisTemplate<String, Object> template = new RedisTemplate<>();template.setConnectionFactory(redisConnectionFactory);// 设置键的序列化方式template.setKeySerializer(new StringRedisSerializer());// 设置值的序列化方式template.setValueSerializer(new GenericJackson2JsonRedisSerializer());return template;}
}
分布式锁工具类
创建一个分布式锁工具类,用于获取和释放锁。
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;import java.util.concurrent.TimeUnit;@Component
public class RedisDistributedLock {@Autowiredprivate RedisTemplate<String, String> redisTemplate;private static final String LOCK_PREFIX = "scheduled_task_lock:";// 锁过期时间(秒)private static final int LOCK_EXPIRE_TIME = 30; /*** 尝试获取分布式锁* @param taskName 任务名称* @param nodeIdentifier 节点标识* @return 是否获取到锁*/public boolean tryAcquireLock(String taskName, String nodeIdentifier) {String lockKey = LOCK_PREFIX + taskName;String lockValue = String.valueOf(System.currentTimeMillis() + LOCK_EXPIRE_TIME * 1000);if (Boolean.TRUE.equals(redisTemplate.opsForValue().setIfAbsent(lockKey, lockValue, LOCK_EXPIRE_TIME, TimeUnit.SECONDS))) {return true;} else {String currentValue = redisTemplate.opsForValue().get(lockKey);if (currentValue != null && Long.parseLong(currentValue) < System.currentTimeMillis()) {String oldValue = redisTemplate.opsForValue().getAndSet(lockKey, lockValue);if (oldValue != null && oldValue.equals(currentValue)) {return true;}}}return false;}/*** 释放分布式锁* @param taskName 任务名称* @param nodeIdentifier 节点标识*/public void releaseLock(String taskName, String nodeIdentifier) {String lockKey = LOCK_PREFIX + taskName;redisTemplate.delete(lockKey);}
}
Token 缓存类
创建一个 Token 缓存类,用于生成、存储和验证 Token,同时提供刷新 Token 的方法。
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;import java.util.concurrent.TimeUnit;@Component
public class TokenCache {private static final String TOKEN_PREFIX = "token:";// Token 过期时间(秒)可以配置化private static final long EXPIRE_TIME = 60 * 60; // 提前 5 分钟刷新 Token可以配置化private static final long REFRESH_THRESHOLD = 60 * 5; @Autowiredprivate RedisTemplate<String, Object> redisTemplate;/*** 生成 Token*/public String generateToken() {return java.util.UUID.randomUUID().toString();}/*** 存储 Token* @param token Token* @param machineId 机器 ID*/public void storeToken(String token, String machineId) {redisTemplate.opsForValue().set(TOKEN_PREFIX + token, machineId, EXPIRE_TIME, TimeUnit.SECONDS);}/*** 验证 Token* @param token Token* @return 验证结果(true 表示有效,false 表示无效)*/public boolean verifyToken(String token) {Object machineId = redisTemplate.opsForValue().get(TOKEN_PREFIX + token);return machineId != null;}/*** 刷新 Token* @param token Token*/public void refreshToken(String token) {redisTemplate.expire(TOKEN_PREFIX + token, EXPIRE_TIME, TimeUnit.SECONDS);}
}
定时任务类
创建一个定时任务类,用于定期检查并刷新即将过期的 Token。使用分布式锁确保同一时间只有一个 Pod 执行该任务。
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;import java.util.Set;@Component
public class TokenRefreshTask {private static final String TASK_NAME = "refreshTokenTask";// 节点标识,可以根据实际情况动态生成private static final String NODE_IDENTIFIER = "node-1"; @Autowiredprivate RedisTemplate<String, Object> redisTemplate;@Autowiredprivate TokenCache tokenCache;@Autowiredprivate RedisDistributedLock redisDistributedLock;/*** 定时任务:刷新即将过期的 Token*/@Scheduled(fixedRate = 60 * 1000) // 每分钟执行一次public void refreshTokenTask() {if (redisDistributedLock.tryAcquireLock(TASK_NAME, NODE_IDENTIFIER)) {try {Set<String> keys = redisTemplate.keys(TokenCache.TOKEN_PREFIX + "*");if (keys != null) {for (String key : keys) {Long ttl = redisTemplate.getExpire(key, TimeUnit.SECONDS);if (ttl != null && ttl <= TokenCache.REFRESH_THRESHOLD) {tokenCache.refreshToken(key.replace(TokenCache.TOKEN_PREFIX, ""));System.out.println("Token 刷新成功: " + key);}}}} finally {redisDistributedLock.releaseLock(TASK_NAME, NODE_IDENTIFIER);}} else {System.out.println("Token 刷新任务已被其他节点执行");}}
}
启动类
确保启动类中启用了定时任务。
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableScheduling;@SpringBootApplication
@EnableScheduling
public class TokenCacheApplication {public static void main(String[] args) {SpringApplication.run(TokenCacheApplication.class, args);}
}
运行结果
在 Kubernetes 环境中部署多个 Pod 后,运行程序,观察输出结果:
Token 刷新成功: token:0f5d3f6e-7f8b-4d9b-8a1b-4d8c7f6e8d9b
Token 刷新任务已被其他节点执行
从结果可以看出,只有获取到分布式锁的 Pod 会执行 Token 刷新任务,其他 Pod 会跳过该任务。
总结
通过引入分布式锁机制,我们成功解决了在 Kubernetes Pod 部署环境下分布式刷新缓存 Token 的问题。使用 Redis 实现的分布式锁确保了同一时间只有一个 Pod 能够执行 Token 刷新任务,避免了重复执行的问题。这种机制不仅适用于 Token 刷新,还可以扩展到其他需要分布式定时任务的场景。
相关文章:
Redis | 基于 Redis 实现机器列表 Token 缓存的 Java 实现
关注:CodingTechWork 引言 在分布式系统中,Token 缓存是一种常见的需求。它可以帮助我们快速验证用户身份,减少对数据库的频繁访问,提高系统的性能和响应速度。本文将介绍如何使用 Redis 来实现机器列表的 Token 缓存,…...
小程序语音识别功能 wx.createInnerAudioContext
页面样式htmlcss <view class"recorder_content"><view class"result_content"><view class"r_title">语音识别结果显示:</view><view class"r_h_input"><text wx:if"{{resultDetails.result}…...
Web网页内嵌福昕OFD版式办公套件实现在线预览编辑PDF、OFD文档
PDF,即Portable Document Format,用于以一种独立于应用程序、硬件、操作系统的方式共享和查看文档;OFD,即Office Open Document Format for Document,是一种在政府公文和法律文件等领域广泛应用的电子文件格式…...
每日一题之日期统计
问题描述 小蓝现在有一个长度为 100100 的数组,数组中的每个元素的值都在 00 到 99 的范围之内。数组中的元素从左至右如下所示: 5 6 8 6 9 1 6 1 2 4 9 1 9 8 2 3 6 4 7 7 5 9 5 0 3 8 7 5 8 1 5 8 6 1 8 3 0 3 7 9 2 7 0 5 8 8 5 7 0 9 9 1 9 4 4 6 …...
ADZS-ICE-2000和AD-ICE2000仿真器在线升级固件
作者的话 近期发现有些兄弟的ICE-2000仿真器链接DSP报错,然后test第四步不通过,我就拿我的仿真器也试了一下,发现ADI悄咪咪的在线升级仿真器固件,有些兄弟不会操作,就会导致仿真器升级失败,连不上目标板&a…...
第十一章:Python PIL库-图像处理
一、PIL库简介 PIL(Python Imaging Library)是一个功能强大的图像处理库,它提供了丰富的图像处理功能,包括图像的打开、处理和保存等操作。PIL支持多种图像文件格式,如JPEG、PNG、BMP等,并且可以完成对图像…...
深入解析缓冲区:计算机世界的“蓄水池”与“加速器”
引言 想象这样一个场景: 你的手机正在播放4K视频,同时下载大型文件 视频画面流畅无卡顿,下载速度稳定在满带宽 但手机的内存只有8GB,下载文件的大小却超过20GB 这看似矛盾的现象背后,缓冲区(Buffer&am…...
Elasticsearch 之 ElasticsearchRestTemplate 聚合查询
前言: 上一篇我们分享了 ElasticsearchRestTemplate 的常用普通查询,本篇我们使用 ElasticsearchRestTemplate 来完成 Elasticsearch 更为复杂的聚合查询。 Elasticsearch 系列文章传送门 Elasticsearch 基础篇【ES】 Elasticsearch Windows 环境安装…...
基础认证-单选题(一)
单选题 1、下列关于request方法和requestlnStream方法说法错误的是(C) A 都支持取消订阅响应事件 B 都支持订阅HTTP响应头事件 C 都支持HttpResponse返回值类型 D 都支持传入URL地址和相关配置项 2、如需修改Text组件文本的透明度可通过以下哪个属性方法进行修改 (C) A dec…...
DeepSeek算法研发闭环解析:如何打造持续进化的AI生产线?
摘要:在AI模型快速迭代的今天,如何构建一个高效、自优化的算法研发体系?DeepSeek通过独特的"数据-训练-评估-部署"闭环架构,实现了AI模型的持续进化。本文将深入剖析其核心设计逻辑与工程实现细节,揭秘支撑千…...
python项目整体文件和依赖打包
python项目整体文件和依赖打包 python项目整体文件和依赖打包 python项目整体文件和依赖打包 准备工作:扫描项目中必要的依赖包 pip install pipreqs pipreqs . 会有一些警告包,需要pip list进行版本修正,这里是三个包第一步:在虚拟环境中安…...
logstash收集数据
防止ES的的I/O的压力过大,使用redis/kafka进行缓冲。 对redis的要求 Redis input plugin | Logstash Reference [8.17] | Elastic 一般企业要求的架构 我实现的架构 filebeat把数据传给logstash 配置好filebeat把收集到的数据输入到redis 然后执行命令࿰…...
智能运维时代的网络拓扑管理:乐维监控的架构可视化实践
在数字化转型的浪潮中,企业IT基础设施正经历着前所未有的复杂化进程。当数以千计的网络设备、服务器、存储系统构成庞大网络体系时,如何实现全局可视化管理已成为企业数字化转型的关键命题。乐维监控网络拓扑系统作为新一代智能运维平台的核心组件&#…...
spring batch 中JpaNamedQueryProvider、JpaNativeQueryProvider两种查询方式对比
完整代码示例:对比两种查询方式 // Employee.java 实体类(包含命名查询) Entity NamedQuery(name "Employee.findAllNamedQuery", query "SELECT e FROM Employee e ORDER BY e.id") // 定义命名查询 public class Em…...
Spring项目中使用EasyExcel实现Excel 多 Sheet 导入导出功能(完整版)
Excel 多 Sheet 导入导出功能完整实现指南 一、环境依赖 1. Maven 依赖 <!-- EasyExcel --> <dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.3.2</version> </dependency>…...
OkHttp的拦截器是如何工作的?
OkHttp 的拦截器是其核心特性之一,它允许开发者在请求和响应的处理过程中插入自定义逻辑。下面为你详细介绍 OkHttp 拦截器的工作原理、分类及执行流程。 拦截器工作原理概述 OkHttp 中的拦截器本质上是实现了Interceptor接口的类。该接口定义了一个intercept方法,在这个方…...
CentOS 7 安装 EMQX (MQTT)
CentOS 7 安装 EMQX 通过 Yum 源安装 EMQX 支持通过 Yum 源安装,您可通过以下 Yum 命令从中自动下载和安装 EMQX。 通过以下命令配置 EMQX Yum 源: curl -s https://assets.emqx.com/scripts/install-emqx-rpm.sh | sudo bash安装以下依赖项ÿ…...
重试机制之指针退避策略算法
一、目的:随着重试次数增加,逐步延长重连等待时间,避免加重服务器负担。 二、计算公式: 每次重试的延迟时间 初始间隔 (退避基数 ^ 重试次数) 通常设置上限防止等待时间过长。 const delay Math.min(initialDelay * Math.pow…...
spring security的过滤器链
Spring Security 的安全功能通过一系列过滤器(Filter)组成的链式结构实现,每个过滤器负责处理特定的安全任务。这些过滤器按特定顺序执行,形成过滤器链(Security Filter Chain)。以下是其核心过滤器及工作原…...
人工智能:officeAI软件,如何调整AI对话界面的字体?
1、首先,随便打开一个excel(使用wps) 依次点击上方的【OfficeAI】—【右侧面板】 2、在弹出的面板中,输入:助手设置 , 然后按【回车】发送出去 3、之后会弹出界面,在【样式设定】中ÿ…...
Qt之共享内存类QSharedMemory的使用及实现原理(全)
目录 1.简介 2.使用 3.实现原理 3.1.Windows内存映射 3.2.POSIX 共享内存 3.3.System V 共享内存 3.4.QSharedMemory的实现原理 4.总结 1.简介 QSharedMemory 是 Qt 框架提供的一个类,用于在不同进程或线程之间实现共享内存的管理。借助共享内存,…...
dockerfile构建镜像方式
在 Docker 中,可使用 docker build 命令依据 Dockerfile 构建镜像。下面为你详细介绍构建镜像的具体方式。 基本构建命令 若要构建镜像,需在包含 Dockerfile 的目录下执行 docker build 命令。基本语法如下: bash docker build -t <镜像…...
Problem A: 接口使用
1.题目问题 2.样例 3.代码实现 补充:注意空格 // 定义Vehicle接口 interface Vehicle {void start();void stop(); }// 实现Vehicle接口的Bike类 class Bike implements Vehicle {Overridepublic void start() {System.out.println("i am bike,i am running&…...
用Python插入Excel表格到Word文档
在日常办公场景中,通过Python脚本自动化整合Excel数据与Word文档,能够实现表格的智能迁移,满足不同场景下数据呈现的专业性要求。直接提取表格内容插入Word适用于需要快速传递核心数据的场景,确保信息精准直达;完整复制…...
合合信息TextIn大模型加速器 2.0来了:智能文档解析和图表解析能力全面升级
合合信息“TextIn大模型加速器 2.0”版本来了:文档解析和图表解析能力全面升级 背景 在日常工作中,我们常常遇到无法直接复制的文档内容或图片内容,这些内容通常需要进行识别和解析。一个典型的例子是,当我们需要将折线图转化为…...
笔记:代码随想录算法训练营day62:108.冗余连接、109.冗余连接II
学习资料:代码随想录 108. 冗余连接 卡码网题目链接(ACM模式) 判断是否有环的依据为,利用并查集,isSame函数,判断当下这条边的两个节点入集前是否为同根,如果是的话,该边就是会构…...
刚刚整理实测可用的股票数据API接口集合推荐:同花顺、雅虎API、智兔数服、聚合数据等Python量化分析各项数据全面丰富
在金融科技高速发展的今天,股票API接口已成为开发者、量化交易者和金融从业者的核心工具之一。它通过标准化的数据接口,帮助用户快速获取实时或历史市场数据,为投资决策、策略回测和金融应用开发提供支持。本文将深入解析股票API的核心功能、…...
消息队列Message Queue
前面,我们在黑点点评中秒杀场景中,首次了解到消息队列MQ,它主要解决了秒杀场景中异步场景,提升了并发性,吞吐量。可是还是对消息队列又很多的疑惑? 消息队列是什么 消息队列是一种通信协议或中间件&#…...
Day 25:股票的最大利润 + 1到n求和
数组 prices 记录了某芯片近期的交易价格,其中 prices[i] 表示的 i 天该芯片的价格。你只能选择 某一天 买入芯片,并选择在 未来的某一个不同的日子 卖出该芯片。请设计一个算法计算并返回你从这笔交易中能获取的最大利润。 如果你不能获取任何利润&…...
如何利用AI智能生成PPT提升工作效率
如何利用AI智能生成PPT提升工作效率?PPT制作曾经是每个人办公生活中的一大痛点。你有多久没有在制作PPT时感到焦头烂额,选模板、调整格式、插入图片,每一项都得花费大量的时间和精力,最后还未必能做出一份令人满意的效果。随着人工…...
