记一次由 jedis 引发的离谱选学问题

背景

我的应用中,使用 jedis 作为连接 redis 的客户端,一直在用的好好的,后来有一个新的组件,也需要使用 redis,但是组件是内部封装的,我只能提供一个 StringReidsTempalte,所以我基于应用本身构造的 factory,又重新构建了一个新的 template。

使用版本

springboot: 3.0.6
jedis: 4.3.2
redis 安装版本: 7.0.11
jdk: 17

现状

当时的现状就是这样的:

有一个统一的factory 构造器,构造一个 factory,并将这个 factory 交由 spring 管理。

应用内部统一使用的缓存客户端,由自定义构造一个 RedisTemplate,而新的组件,使用 factory 构造一个新的 StringRedisTemplate。

表象

我在本地启动的时候,没有任何问题,正常和 redis 交互。

部署到服务器启动的时候,会在实际和 redis 交互的底层逻辑处,报出异常。

报错位置1:

org.springframework.data.redis.connection.jedis.JedisStringCommands#set(byte[], byte[], org.springframework.data.redis.core.types.Expiration, org.springframework.data.redis.connection.RedisStringCommands.SetOption)

img在这个地址执行的时候,invoke 方法没问题,执行到 from 的时候,debug 执行,会报上面构建的 SetParams 这个类找不到。(只有 debug 会报,正常执行该堆栈打印不出)

通过解包,找到实际服务器的 jar 包中的 lib,发现对应的jar 包是有的,也存在这个类。

后面我就不管这个,果然就继续执行了,这个地方并无卡点。

执行到 getOrElse 的时候,就会报另外一个异常,也就是服务器会打印出来的异常,Unknown redis exception.

堆栈信息中,大概得处理逻辑就是从cgLib 中获取客户端的时候,会有一些地方有卡点,不过太过深入,没有继续追,肯定不是底层逻辑的 bug,只能是我的客户端构造的有问题。

于是我就开始仔细看我构造出来的客户端,和我正常的客户端比较。

然而这个也没有什么异常的情况。

我的自己构建的客户端连接池,正常使用,用了很久了,一直没问题,实际到底层的时候,都是在一个地方使用的,应该是没啥问题。

通过网上找关于 jedis 客户端的使用 demo,几乎和我的没差别,只是调优参数不一样。

jedis 报错:Unknown redis exception 的几种常见问题

  1. Redis 服务未启动或者未连接:在程序连接 Redis 时,如果 Redis 服务未启动或者程序无法连接到 Redis,就会出现这个异常。确保 Redis 服务已启动,且连接信息配置正确。
  2. Redis 服务故障:Redis 服务本身发生故障导致连接断开,也会引起这个异常。这时可以检查 Redis 服务是否正常,以及网络连接是否正常。
  3. Redis 客户端版本问题:Spring Data Redis 依赖于 lettuce 和 Jedis 客户端来访问 Redis。如果使用了不兼容的客户端版本,就会出现这个异常。确保 Spring Data Redis、lettuce 和 Jedis 的版本匹配。
  4. Redis 操作出错:使用 Redis 进行操作时,可能会遇到一些错误情况,如 key 不存在,操作类型不匹配等。这时检查 Redis 的操作是否合规,可以帮助解决这个异常。

针对这些可能产生该异常的情况,可以分别进行如下处理:

  1. 确认 Redis 服务已启动,并检查网络连接和连接信息是否正确。
  2. 检查 Redis 服务是否正常,以及网络连接是否正常。
  3. 选择适合的 Spring Data Redis、lettuce 和 Jedis 版本,确保版本匹配。
  4. 确认 Redis 的操作是否合规,以及检查 Redis key 是否存在。

解决问题

最终发现,根本无法下手,因为我本地是好的,java 版本都没问题。

最终尝试使用 lettuce。

问题解决,正常使用。

如果有各位大佬知道具体原因,欢迎交流!不胜感激!

拓展

lettuce 和 jedis比较

spring boot 本身默认使用的就是 lettuce,所以个人觉得直接使用 lettuce 会更好一些。

lettuce 是 jedis 的后起之秀,相较于 jedis 来说,lettuce 有如下优点

  • 线程安全(jedis 线程非安全)
  • 基于 Netty 框架的事件驱动通信,可以异步(jedis 为同步阻塞 IO,不支持异步)
  • 基于异步+线程安全,所以更适合分布式缓存。
    • ps:我的应用就是分布式的,考虑到有可能是线程安全导致的问题,所以尝试使用 lettuce,结果就解决问题了。(只是猜测可能是线程安全问题导致,并没有实际验证,虽然解决,可能只是误打误撞)

相较于 jedis,lettuce 的缺点也有

  • jedis 提供了更为全面的 reidis 操作特性的 api。
  • jedis 的 api 基本和 redis 的指令一一对应,使用简单,更容易理解。而 lettuce 的api 更抽象一些,学习成本会更高一些。

lettuce 使用

和 jedis 相比,lettuce 使用更简单,因为 lettuce 并不需要配置连接池,因为 lettuce 单链接的性能就很好,线程池数量太低,会导致性能降低,太高和单链接性能差异不大,但是资源消耗更多。

所以简单配置即可。

spring boot 3之后,也移除了直接配置连接池的入口。只需要简单配置 redis 的链接方式即可。

    @Bean("lettuceConnectionFactory")public RedisConnectionFactory lettuceConnectionFactory(CacheConfiguration configuration) {logger.info("开始构建 redis factory...{}", JedisConnectionFactory.class.getName());
//        GenericObjectPoolConfig<RedisProperties.Lettuce> poolConfig = getLettucePoolConfig();
//
//        LettuceClientConfiguration clientConfig = LettuceClientConfiguration.builder()
//                .build();RedisStandaloneConfiguration redisConfig = getRedisConfiguration(configuration);
//        LettuceConnectionFactory factory = new LettuceConnectionFactory(redisConfig, clientConfig);LettuceConnectionFactory factory = new LettuceConnectionFactory(redisConfig);factory.afterPropertiesSet();return factory;}

注意:lettuce 的 factory 创建完成之后,一定要调用afterPropertiesSet方法,否则在实际使用的时候就会报错。

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

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

相关文章

Java 多线程之 LockSupport (阻塞和唤醒线程)

文章目录 一、概述二、使用方法三、测试示例1四、测试示例2 一、概述 LockSupport 是Java并发包中的一个工具类&#xff0c;用于线程的阻塞和唤醒。它提供了一种基于线程的许可&#xff08;permit&#xff09;的方式来实现线程的阻塞和唤醒&#xff0c;而不需要显式地使用锁。例…

【无线网络技术】——无线广域网(学习笔记)

&#x1f4d6; 前言&#xff1a;无线广域网(WWAN)是指覆盖全国或全球范围内的无线网络&#xff0c;提供更大范围内的无线接入&#xff0c;与无线个域网、无线局域网和无线城域网相比&#xff0c;它更加强调的是快速移动性。典型的无线广域网&#xff1a;蜂窝移动通信系统和卫星…

Linux UUCP命令教程:如何在Linux系统中进行文件复制(附实例详解和注意事项)

Linux UUCP命令介绍 UUCP&#xff08;Unix-to-Unix Copy&#xff09;是一套允许远程执行命令和传输文件的程序。UUCP命令是该套件中的一个程序&#xff0c;它为请求文件复制操作提供了用户界面。UUCP套件还包括uux&#xff08;远程命令执行的用户界面&#xff09;、uucico&…

Java期末复习题之抽象类、接口

点击返回标题->23年Java期末复习-CSDN博客 第1题. 首先设计一个学生抽象类Student&#xff0c;其数据成员有name(姓名)、age(年龄)和degree(学位)&#xff0c;以及一个抽象方法show()。然后由Student类派生出本科生类Undergraduate和研究生类Graduate&#xff0c;本科生类Un…

js moment计算当前时间到24:00:00的剩余时间

2023.12.7今天我学习了如何计算当前的时间到24:00:00剩下的时间&#xff0c;https://momentjs.cn/ const now moment(); // 获取当前时间const endOfDay moment().endOf(day); // 设置当天的 23:59:59const duration moment.duration(endOfDay.diff(now)); // 计算剩余时间的…

第 7 部分 — 增强 LLM 安全性的策略:数学和伦理框架

一、说明 增强大型语言模型 (LLM) 安全性的追求是技术创新、道德考虑和实际应用的复杂相互作用。这项努力需要一种深入而富有洞察力的方法&#xff0c;将先进的数学模型与道德原则和谐地融合在一起&#xff0c;以确保LLM的发展不仅在技术上稳健&#xff0c;而且在道德上合理且对…

C#winform点击按钮下载数据库中表的字段到Excel上

C#winform点击按钮下载数据库中表的字段到Excel上 需求&#xff1a;C#winform点击按钮下载数据库中表的字段到Excel&#xff0c;并计算下载消耗的时间以及文件存放位置。 C#实现 using System; using System.Data; using System.Data.OleDb; using System.Data.SqlClient; u…

Flutter 如何更新showModalBottomSheet 中的数据

showDialog(context: context,builder: (context) {String label test;//StatefulBuilderreturn StatefulBuilder(//在这里为了区分&#xff0c;在构建builder的时候将setState方法命名为了setDialogState。builder: (context, setDialogState) {print(label $label);return …

【LeetCode】268. 丢失的数字

268. 丢失的数字 难度&#xff1a;简单 题目 给定一个包含 [0, n] 中 n 个数的数组 nums &#xff0c;找出 [0, n] 这个范围内没有出现在数组中的那个数。 示例 1&#xff1a; 输入&#xff1a;nums [3,0,1] 输出&#xff1a;2 解释&#xff1a;n 3&#xff0c;因为有 3…

[Makefile] include 关键字

在 Makefile 中&#xff0c;include 关键字的作用是引入其他文件的内容&#xff0c;通常用于将其他 Makefile 文件&#xff08;通常是头文件&#xff09;的内容包含到当前的 Makefile 中。这样可以实现模块化管理和代码重用。 include使用 使用 include 关键字的语法如下&…

网络攻击(一)--安全渗透简介

1. 安全渗透概述 目标 了解渗透测试的基本概念了解渗透测试从业人员的注意事项 1.1. 写在前面的话 在了解渗透测试之前&#xff0c;我们先看看&#xff0c;信息安全相关的法律是怎么样的 中华人民共和国网络安全法 《中华人民共和国网络安全法》由全国人民代表大会常务委员会…

Spring Cloud切换内嵌Tomcat为宝兰德Application Server

目录 替换Tomcat中间件Tomcat是什么Spring Cloud剔除tomcat引入宝兰德Application Server打包运行授权导入 替换Tomcat中间件 Tomcat是什么 Spring Cloud剔除tomcat <!--集成springmvc框架 --><dependency><groupId>org.springframework.boot</groupId&…

Boost:asio多io_service,多线程run

多io_service,多线程run,相当于多个线程分别处理注册在不同io_service上的回调,也就是每个线程排某个io_service的异步处理: //mio_mth.cpp #include <boost/asio.hpp> #include <boost/date_time/posix_time/posix_time_types.hpp> #include <iostream>…

MAC PHP版本安装问题

安装php 7.4版本不成功 Error: php7.4 has been disabled because it is a versioned formula! 因为php7.4官方已经不再维护&#xff0c;所以Hombrew将该php版本移出了repository&#xff0c;所以安装不了。 解决办法 从第三方仓库中安装 //将第三方仓库加入brew brew tap sh…

7.1 C++11指针空值—nullptr

一、NULL和nullptr区别 NULL是宏定义&#xff0c;nullptr是关键字。 #ifndef NULL#ifdef __cplusplus#define NULL 0#else#define NULL ((void *)0)#endif #endif nullptr可以隐式转换为任意指针类型&#xff0c;而NULL需要显示转换 void func(char *) {std::cout <<…

Java安全之Commons Collections6分析

CC6分析 import org.apache.commons.collections.*; import org.apache.commons.collections.functors.ChainedTransformer; import org.apache.commons.collections.functors.ConstantTransformer; import org.apache.commons.collections.functors.InvokerTransformer; impo…

Flask 动态路由、请求数据接收、视图函数返回值详解

一、动态路由 在前面的博客中&#xff0c;我们学习了如何创建基本的 Flask 应用&#xff0c;并定义了一些简单的路由。但有时候&#xff0c;我们需要更加灵活的路由&#xff0c;能够根据用户请求的不同来动态生成响应。Flask 提供了动态路由的功能&#xff0c;使我们能够轻松处…

上网监控软件——安全与隐私的平衡

网络已经成为人们生活和工作中不可或缺的一部分。然而&#xff0c;随着网络使用的普及&#xff0c;网络安全问题也日益突出。上网监控软件作为网络安全领域的一个重要组成部分&#xff0c;在保护企业和家庭网络安全方面发挥着重要作用。 本文将探讨上网监控软件的背景、功能、优…

我的Android播放器封装经验

近段时间&#xff0c;电视家不能用了&#xff0c;好吧&#xff0c;自己开发一个APP。其实也不是开发&#xff0c;而是基于现有的播放器核心自己封装一个&#xff0c;只要能够非常方便操作观看电视就好。 当然&#xff0c;前提是要有节目源&#xff0c;这个我早已完成&#xff…

1-2算法基础-常用库函数

1.排序 sort(first,last,cmp) first指向要排序范围的第一个元素&#xff0c;从0起 last指向要排序范围的最后一个元素的下一个位置 cmp&#xff08;可选&#xff09;&#xff0c;自定义函数&#xff0c;默认从小到大 评测系统 #include <iostream> #include<algorith…