springboot + redis多数据源 + jedis集群模式

2023-10-27

最近有个项目需要redis支持多个集群,网上搜了下,发现有个开源的项目spring-boot-starter-dynamic-redis,代码写的挺好,可惜只有单机版的,于是fork了他的代码改了下,支持jedis集群模式。

新代码昨天已提交给了原作者,还没看到他的回复。今天就先写个博客简单介绍下吧。

支持集群版的项目地址fork from spring-boot-starter-dynamic-redis

客户端集成方式如下。

1.在pom.xml中添加如下依赖:

<dependency>
    <groupId>com.mk</groupId>
    <artifactId>spring-boot-starter-dynamic-redis</artifactId>
    <!-- 单机版 1.0-SNAPSHOT-->
    <!-- <version>1.0-SNAPSHOT</version> -->
    
    <!-- 集群版 1.1-SNAPSHOT-->
    <version>1.1-SNAPSHOT</version>
</dependency>

2.在yml配置文件中添加如下配置:

dms:
  dynamic:
    redis:
      enabled: true
      #是否启用集群版配置
      isJedisCluster: true
      connection:
        # 第一个Redis连接
        redis1:
          #host单机版填单个ip值; 集群版填ip和端口号,多个用逗号分隔: ip1:port1,ip2:port2,...; 
          host: 192.168.0.1:6573,192.168.0.2:6573,192.168.0.3:6573
          #单机版填端口, 集群版忽略不用填
          port: 6379
          database: 1
          timeout: 3000
          jedis:
            pool:
              max-wait: 3000
              max-active: 100
              max-idle: 20
              min-idle: 0
        # 第二个Redis连接
        redis2:
          host: 127.0.0.1
          port: 6379
          database: 2
          timeout: 3000
          jedis:
            pool:
              max-wait: 3000
              max-active: 100
              max-idle: 20
              min-idle: 0

3.在启动类上面添加@ComponentScan注解,如下:

//@EnableDynamicRedis //此注解貌似没啥用
@SpringBootApplication
@ComponentScan({"com.mk.*"})
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class);
    }
}

4.1单机版,创建Redis配置类,如下:

@Configuration
public class RedisConfig {

    /**
     * 是否启用jedis集群模式,默认为否
     */
    @Value("${dms.dynamic.redis.isJedisCluster:false}")
    private boolean isJedisCluster;
    
    @Resource
    private DynamicRedisProvider dynamicRedisProvider;

    @Bean(name = "demo1Redis")
    public RedisTemplate demo1Redis() {
         return new StringRedisTemplate(dynamicRedisProvider.loadRedis(isJedisCluster).get("redis1"));
    }

    @Bean(name = "demo2Redis")
    public RedisTemplate demo2Redis() {
        return new StringRedisTemplate(dynamicRedisProvider.loadRedis(isJedisCluster).get("redis2"));
    }
}

4.2.单机版测试类:

@RestController
public class TestController {
    @Resource(name = "demo1Redis")
    private StringRedisTemplate demo1Template;

    @Resource(name = "demo2Redis")
    private StringRedisTemplate demo2Template;

    @GetMapping("/testRedis")
    public String testRedis(){
        demo1Template.opsForValue().set("testKey", "我存放到Redis下db为1的库");
        demo2Template.opsForValue().set("testKey", "我存放到Redis下db为2的库");
        return "success";
    }
}

5.1集群版,创建Redis配置类,如下:

@Configuration
@ConditionalOnProperty(prefix = DynamicRedisProperties.PREFIX, value = "enabled", matchIfMissing = false)
public class CustomerDynamicRedisConfig extends DynamicRedisConfig {
    /**
     * 是否启用jedis集群模式,默认为否
     */
    @Value("${dms.dynamic.redis.isJedisCluster:false}")
    private boolean isJedisCluster;

    /**
     * 指定序列化方式: key <== String, value <== json
     */
    @Bean(name = RedisHelper.REDIS_1)
    public RedisTemplate<Object, Object> redis1() {
        return createRedisTemplate(dynamicRedisProvider.loadRedis(isJedisCluster).get(RedisHelper.REDIS_1));
    }

    /**
     * 指定序列化方式: key <== String, value <== String
     */
    @Bean(name = RedisHelper.STR_REDIS_1)
    public RedisTemplate<String, String> stringRedis1() {
        return createStringRedisTemplate(dynamicRedisProvider.loadRedis(isJedisCluster).get(RedisHelper.REDIS_1));
    }

    @Bean(name = RedisHelper.REDIS_2)
    public RedisTemplate<Object, Object> redis2() {
        return createRedisTemplate(dynamicRedisProvider.loadRedis(isJedisCluster).get(RedisHelper.REDIS_2));
    }

    @Bean(name = RedisHelper.STR_REDIS_2)
    public RedisTemplate<String, String> stringRedis2() {
        return createStringRedisTemplate(dynamicRedisProvider.loadRedis(isJedisCluster).get(RedisHelper.REDIS_2));
    }
}

5.2. 集群版RedisHelper工具类:

/**
 * 获取对应redis操作的帮助类。
 *
 * @author machunlin@126.com
 */
public class RedisHelper {
    private static final Map<String, RedisService> map = new ConcurrentHashMap<>();

    public static final String REDIS_1 = "redis1";
    public static final String STR_REDIS_1 = "stringRedis1";

    public static final String REDIS_2 = "redis2";
    public static final String STR_REDIS_2 = "stringRedis2";

    /**
     * redis value值为json序列化
     */
    public static RedisService getRedis1() {
        if (map.containsKey(REDIS_1)) {
            return map.get(REDIS_1);
        }

        RedisService _redisService = new RedisService();
        RedisTemplate redisTemplate1 = (RedisTemplate) SpringContextUtil.getBean(REDIS_1);
        _redisService.setRedisTemplate(redisTemplate1);
        map.put(REDIS_1, _redisService);
        return _redisService;
    }

    /**
     * redis value值为string序列化
     */
    public static RedisService getStringRedis1() {
        if (map.containsKey(STR_REDIS_1)) {
            return map.get(STR_REDIS_1);
        }

        RedisService _redisService = new RedisService();
        StringRedisTemplate strRedisTemplate1 = (StringRedisTemplate) SpringContextUtil.getBean(STR_REDIS_1);
        _redisService.setRedisTemplate(strRedisTemplate1);
        map.put(STR_REDIS_1, _redisService);
        return _redisService;
    }

    public static RedisService getRedis2() {
        if (map.containsKey(REDIS_2)) {
            return map.get(REDIS_2);
        }

        RedisService _redisService = new RedisService();
        RedisTemplate redisTemplate = (RedisTemplate) SpringContextUtil.getBean(REDIS_2);
        _redisService.setRedisTemplate(redisTemplate);
        map.put(REDIS_2, _redisService);
        return _redisService;
    }

    public static RedisService getStringRedis2() {
        if (map.containsKey(STR_REDIS_2)) {
            return map.get(STR_REDIS_2);
        }

        RedisService _redisService = new RedisService();
        StringRedisTemplate strRedisTemplate1 = (StringRedisTemplate) SpringContextUtil.getBean(STR_REDIS_2);
        _redisService.setRedisTemplate(strRedisTemplate1);
        map.put(STR_REDIS_2, _redisService);
        return _redisService;
    }

}

5.3.集群版测试类

@RestController
@RequestMapping(value = "/redis")
public class RedisTestController {

    @GetMapping(value = "/testSet")
    public String testSet(String redisName, String key) {
        RedisService redisService;
        if ("1".equals(redisName)) {
            redisService = RedisHelper.getRedis1();
        } else if ("2".equals(redisName)) {
            redisService = RedisHelper.getRedis2();
        } else if ("str1".equals(redisName)) {
            redisService = RedisHelper.getStringRedis1();
        } else {
            throw new IllegalArgumentException("redisName错误");
        }
        redisService.set(key, "hahahah你好你好", 600L);
        return (String) redisService.get(key);
    }

    @GetMapping(value = "/testGet")
    public String testGet(String redisName, String key) {
        RedisService redisService;
        if ("1".equals(redisName)) {
            redisService = RedisHelper.getRedis1();
        } else if ("2".equals(redisName)) {
            redisService = RedisHelper.getRedis2();
        } else if ("str1".equals(redisName)) {
            redisService = RedisHelper.getStringRedis1();
        } else {
            throw new IllegalArgumentException("redisName错误");
        }
        return (String) redisService.get(key);
    }
}

再次感谢原作者开源贡献的代码!

原项目(单机版)
集群版

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

springboot + redis多数据源 + jedis集群模式 的相关文章

  • ElasticBeanstalk Java,Spring 活动配置文件

    我正在尝试通过 AWS ElasticBeanstalk 启动 spring boot jar 一切正常 配置文件为 默认 有谁知道如何为 java ElasticBeanstalk 应用程序 不是 tomcat 设置活动配置文件 spri
  • AES 加密 Java/plsql

    我需要在Java和plsql DBMS CRYPTO for Oracle 10g 上实现相同的加密 解密应用程序 两种实现都工作正常 但这里的问题是我对相同纯文本的加密得到了不同的输出 下面是用于加密 解密过程的代码 Java 和 PLS
  • Android Studio 在编译时未检测到支持库

    由于 Android Studio 将成为 Android 开发的默认 IDE 因此我决定将现有项目迁移到 Android studio 中 项目结构似乎不同 我的项目中的文件夹层次结构如下 Complete Project gt idea
  • Redis+Docker+Django - 错误 111 连接被拒绝

    我正在尝试使用 Redis 作为使用 Docker Compose 的 Django 项目的 Celery 代理 我无法弄清楚我到底做错了什么 但尽管控制台日志消息告诉我 Redis 正在运行并接受连接 事实上 当我这样做时 docker
  • CXF Swagger2功能添加安全定义

    我想使用 org apache cxf jaxrs swagger Swagger2Feature 将安全定义添加到我的其余服务中 但是我看不到任何相关方法或任何有关如何执行此操作的资源 下面是我想使用 swagger2feature 生成
  • Convert.FromBase64String 方法的 Java 等效项

    Java 中是否有相当于Convert FromBase64String http msdn microsoft com en us library system convert frombase64string aspx which 将指
  • 如何在jsp代码中导入java库?

    我有以下jsp代码 我想添加 java io 等库 我怎样才能做到这一点
  • Prim 的迷宫生成算法:获取相邻单元格

    我基于 Prim 算法编写了一个迷宫生成器程序 该算法是 Prim 算法的随机版本 从充满墙壁的网格开始 选择一个单元格 将其标记为迷宫的一部分 将单元格的墙壁添加到墙壁列表中 While there are walls in the li
  • 序列化对象以进行单元测试

    假设在单元测试中我需要一个对象 其中所有 50 个字段都设置了一些值 我不想手动设置所有这些字段 因为这需要时间而且很烦人 不知何故 我需要获得一个实例 其中所有字段都由一些非空值初始化 我有一个想法 如果我要调试一些代码 在某个时候我会得
  • Java中接口作为方法参数

    前几天去面试 被问到了这样的问题 问 反转链表 给出以下代码 public class ReverseList interface NodeList int getItem NodeList nextNode void reverse No
  • 尝试使用 Ruby Java Bridge (RJB) gem 时出现错误“无法创建 Java VM”

    我正在尝试实现 Ruby Java Bridge RJB gem 来与 JVM 通信 以便我可以运行 Open NLP gem 我在 Windows 8 上安装并运行了 Java 所有迹象 至少我所知道的 都表明 Java 已安装并可运行
  • org.jdesktop.application 包不存在

    几天以来我一直在构建一个 Java 桌面应用程序 一切都很顺利 但是今天 当我打开Netbeans并编译文件时 出现以下编译错误 Compiling 9 source files to C Documents and Settings Ad
  • Java中未绑定通配符泛型的用途和要点是什么?

    我不明白未绑定通配符泛型有什么用 具有上限的绑定通配符泛型 stuff for Object item stuff System out println item Since PrintStream println 可以处理所有引用类型 通
  • Tomcat 6找不到mysql驱动

    这里有一个类似的问题 但关于类路径 ClassNotFoundException com mysql jdbc Driver https stackoverflow com questions 1585811 classnotfoundex
  • Windows 上的 Nifi 命令

    在我当前的项目中 我一直在Windows操作系统上使用apache nifi 我已经提取了nifi 0 7 0 bin zip文件输入C 现在 当我跑步时 bin run nifi bat as 管理员我在命令行上看到以下消息 但无法运行
  • 如何修复“sessionFactory”或“hibernateTemplate”是必需的问题

    我正在使用 Spring Boot JPA WEB 和 MYSQL 创建我的 Web 应用程序 它总是说 sessionFactory or hibernateTemplate是必需的 我该如何修复它 我已经尝试过的东西 删除了本地 Mav
  • KeyPressed 和 KeyTyped 混淆[重复]

    这个问题在这里已经有答案了 我搜索过之间的区别KeyPressedand KeyTyped事件 但我仍然不清楚 我发现的一件事是 Keypressed 比 KeyTyped 首先被触发 请澄清一下这些事件何时被准确触发 哪个适合用于哪个目的
  • 中断连接套接字

    我有一个 GUI 其中包含要连接的服务器列表 如果用户单击服务器 则会连接到该服务器 如果用户单击第二个服务器 它将断开第一个服务器的连接并连接到第二个服务器 每个新连接都在一个新线程中运行 以便程序可以执行其他任务 但是 如果用户在第一个
  • java8 Collectors.toMap() 限制?

    我正在尝试使用java8Collectors toMap on a Stream of ZipEntry 这可能不是最好的想法 因为在处理过程中可能会发生异常 但我想这应该是可能的 我现在收到一个我不明白的编译错误 我猜是类型推理引擎 这是
  • Swagger/Openapi-Annotations:如何使用 $ref 生成 allOf?

    我正在生成 Rest 端点 包括添加OpenAPI Swagger对生成的代码进行注释 虽然它对于基本类型运行得很好 但我在自定义类方面遇到了一些问题 现在我有很多自定义类的重复架构条目 使用 Schema 实现 MyClass class

随机推荐

  • CloudCompare——计算点云曲率

    目录 1 找到曲率计算功能 2 设置计算参数 3 可视化曲率计算结果 4 保存计算结果 5 完整操作流程 6 相关链接 1 找到曲率计算功能 2 设置计算参数 只有一个参数 位置处用于查找最近邻点的球邻域半径 3 可视化曲率计算结果 4 保
  • 保卫家园(小白版)

    保卫家园 牛客 题目链接 https ac nowcoder com acm problem 205068 题目 题目描述 为了抵御深渊的蔓延 被深渊毁掉家园的人们组建法兰不死队来镇压深渊 已知法兰不死队的最大编制为k 即队伍最多能有k人
  • (详解)用C语言实现一个能够连续展开的扫雷。

    功能介绍 我们实现一个扫雷 那么必须具有其基本的功能 如下图 那么相比较一般的用C实现的扫雷 我们需要添加如下的三个功能 防止第一次被炸死 标记雷 连续展开 代码实现以及解释 一 主函数的实现以及扫雷的构思框架 test c int mai
  • 自举电容的工作原理-3句话总结

    1 一道面试题 照例 先抛出来一道面试题 Buck电路的SW引脚通常会放置一颗小电容 这颗电容有什么用 这个问题问得比较细 非常考验被面试者的硬件基本功 2 似懂非懂的答案 有小伙伴可能不太清楚这颗电容的作用 回答不上来 也有小伙伴脱口而出
  • 1 shell备份数据库MYSQL案例

    SHELL编程实战MYSQL备份脚本 编程思路 1 备份MYSQL数据库 备份的工具的选择 数据量小于100G使用全备 基于mysqldump工具备份 备份会锁表 数据量大于100G使用增量备份 基于xtrabackup工具备份 Innod
  • STM32CubeMX新建工程并点亮一个LED

    可提前看 STM32CubeMX环境安装 保姆级 目录 进入STM32CubeMX界面 双击软件 可能会出现的弹窗 更改固件路径 新建工程 进入配置环境 以一个点灯程序为例 GPIO配置 输出电平设置 输出模式 上下拉 输出速度 User
  • 方差

    什么是随机变量的方差 方差反映随机变量取值的什么性质 方差是一个常用来体现随机变量的取值分散程度的量 设 X X X是一个随机变量 若 E X
  • (c语言)输出Programming is fun!

    题目要求 输入 无 输出 Programming is fun 具体代码如下 include
  • 打印从1到最大的n位数

    链接 打印从1到最大的n位数 牛客题霸 牛客网 nowcoder com 描述 输入数字 n 按顺序打印出从 1 到最大的 n 位十进制数 比如输入 3 则打印出 1 2 3 一直到最大的 3 位数 999 1 用返回一个整数列表来代替打印
  • Hands-On Hyperledger Fabric——Fabric的证书(账号)体系

    文章目录 Fabric证书是什么 Fabric证书服务器 Fabric ca fabric ca server的初始化 fabric ca server 配置文件 fabric ca client 使用 将Fabric ca server绑
  • GPT分区规划与各分区作用解析

    1 GPT分区规划 安装用EFI引导的Windows 10前 手动为硬盘分区 推荐方案如下 序号 分区名 起始柱面 磁头 扇区 容量 MBR保留扇区 GPT分区表 1MB 0 ESP分区 0 32 33 96MB 空白 预留给扩展ESP时使
  • 【H5】 svg画贝塞尔曲线方法

    H5 svg画贝塞尔曲线方法 d属性M 起始坐标 L 结束坐标 H 水平线 V 垂直线 A 圆弧 Z 闭合路劲 C S Q T贝塞尔曲线大写为绝对坐标 具体的坐标位置 小写为相对坐标 相对起始坐标点的具体长度 A命令x 径y半径角度弧长 0
  • 关于虚拟机.vmdk与.ovf 磁盘装载问题

    与 vmdk磁盘装载相关的两种方式 0 前言 1 只有 vmdk文件 2 带有 ovf vmdk文件 0 前言 提一嘴 现在用的比较多的虚拟机创建或者拷贝方式有两种 第一种是iso光盘映像装载 第二种是OVF导入 分别为 ISO的装载方式最
  • build JAX from source code

    Building from source JAX documentation
  • visual studio:不能加载.vdproj

    总结 下载后关闭所有vs项目 重新打开目标工程 需要完成次扩展的后续安装任务 参考 https www cnblogs com hofmann p 11183457 htm
  • 122FPS,51.9mAP 超轻量关键点检测算法PP-TinyPose来啦!

    在人机交互场景中 机器可以识别人的手势 肢体动作 表情 你可知背后的核心技术是什么吗 没错 就是关键点检测技术 它能帮你实现精准的人机交互任务 如手势控制 智能健身 体感游戏等 还可以识别交通违规 打架斗殴 违规操作等异常行为 话不多说 我
  • DVWA——XSS(Dom low&medium)

    此文章只用于学习 请勿用作其他违法犯罪行为 以下部分文字内容以图片形式展示 因为JS代码打不上去 目录 前言 XSS攻击流程 XSS的危害 XSS的漏洞类型 JS基本语句 XSS Dom Low XSS Dom Medium 前言 XSS被
  • Ubantu扩展SWAP区,使用gparted,以及死机非热启动解决方法

    Ubantu扩展SWAP区 使用gparted 以及死机非热启动解决方法 Swap分区 强制重启 扩充SWAP Swap分区 Swap分区是用来扩展内存的 即使用一部分硬盘空间作为交换 个人认为电脑内存大于16G即不需要分配Swap空间 我
  • Python字符串替换的3种方法

    Python字符串替换笔记主要展示了如何在Python中替换字符串 Python中有以下几种替换字符串的方法 本文主要介绍前三种 replace方法 常用 translate方法 re sub方法 字符串切片 根据Python字符串切片方法
  • springboot + redis多数据源 + jedis集群模式

    最近有个项目需要redis支持多个集群 网上搜了下 发现有个开源的项目spring boot starter dynamic redis 代码写的挺好 可惜只有单机版的 于是fork了他的代码改了下 支持jedis集群模式 新代码昨天已提交