jedis+redis pipeline诡异的链接损坏、数据读取异常问题解决

文章目录

问题现象

栈溢出(不断的重连)

shell 复制代码
		at redis.clients.jedis.Connection.sendCommand(Connection.java:163)
        at redis.clients.jedis.Connection.sendCommand(Connection.java:154)
        at redis.clients.jedis.BinaryClient.auth(BinaryClient.java:815)
        at redis.clients.jedis.BinaryClient.connect(BinaryClient.java:145)
        at redis.clients.jedis.Connection.sendCommand(Connection.java:163)
        at redis.clients.jedis.Connection.sendCommand(Connection.java:154)
        at redis.clients.jedis.BinaryClient.auth(BinaryClient.java:815)
        at redis.clients.jedis.BinaryClient.connect(BinaryClient.java:145)

读取超时

未知响应

尝试读取损坏的链接

读取到的数据和自己要读的无关,导致空指针、类型转换错误,数据读取错乱



问题写法

对Redis操作封装了一个Redis类

同事对pipeline加了这个方法

使用的时候

java 复制代码
   public Map<String, AnswerModel> batchGetAnswersPipeline(List<String> answerIdList) {
        Map<String, AnswerModel> resultMap = new HashMap<>();
        try (Pipeline pipelined = redis.pipelined()) {
            List<Response<Map<String, String>>> responses = new ArrayList<>();
            for (String answerId : answerIdList) {
                String key = String.format(MOMENT_ANSWER_INFO, answerId);
                responses.add(pipelined.hgetAll(key));
            }
            pipelined.sync();
            for (Response<Map<String, String>> response : responses) {
                Map<String, String> map = response.get();
                AnswerModel answers = BeanUtil.mapToBean(map, AnswerModel.class);
                if (answers != null){
                    resultMap.put(answers.getAnswerId(), answers);
                }
            }
        } catch (Exception e) {
            log.error("执行batchGetAnswersPipeline发生异常, redis pipeline error", e);
            throw new CatVillageException();
        }
        return resultMap;
    }

问题分析

在redis这个工具类中调pipelined时获取一个Pinelined对象,而获取这个方法里用了try with ,with了一个jedis链接,当try结束时链接会被归还jedispool连接池。而返回的这个pipeline仍然在用这个链接,当其他线程去拿链接的时候可能拿到的正好是这个链接,导致多个线程共用一个链接,一个线程在执行pipeline多条命令,另一个线程也在用这个链接。而jedis底层执行命令的时候是使用的OutputStream流式去执行命令,使用二进制流读取结果。当出现这种线程不安全问题的时候,读取写入就会有问题。

修复

删除Redis工具类中的pipelined方法,使用getJedis方法获取链接对象,再获取管道Pineline对象,自行close管道,最后执行完释放管道,链接。

java 复制代码
public Map<String, QuestionModel> batchGetQuestionPipeline(List<String> questionIdList) {
        if (CollectionUtils.isEmpty(questionIdList)){
            return new HashMap<>();
        }
        Map<String, QuestionModel> resultMap = new HashMap<>();
        try (Jedis jedis = redis.getJedis()) {
            Pipeline pipelined = jedis.pipelined();
            List<Response<Map<String, String>>> responses = new ArrayList<>();

            for (String questionId : questionIdList) {
                String key = String.format(MOMENT_QUESTION_INFO, questionId);
                responses.add(pipelined.hgetAll(key));
            }

            pipelined.sync();

            for (Response<Map<String, String>> response : responses) {
                Map<String, String> map = response.get();
                QuestionModel question = convertFromMap(map);
                if (question != null) {
                    resultMap.put(question.getPostId(), question);
                }
            }
            pipelined.close();
        } catch (Exception e) {
            log.error("执行batchGetQuestionPipeline发生异常, redis pipeline error", e);
            throw new CatVillageException();
        }

        return resultMap;
    }

注意点

若jedis在3.7版本以下的版本也会有管道二进制读取异常问题,请升级到3.7.0+版本

相关推荐
半夏知半秋8 分钟前
基于跳跃表的zset实现解析(lua版)
服务器·开发语言·redis·学习·lua
二掌柜,酒来!7 小时前
完美解决:应用版本更新,增加字段导致 Redis 旧数据反序列化报错
redis·spring·bootstrap
小熊h10 小时前
redis 集群——redis cluster(去中心化)
redis·去中心化
得意霄尽欢11 小时前
Redis之底层数据结构
数据结构·数据库·redis
吐泡泡_12 小时前
Redis(集群)
redis
爱吃烤鸡翅的酸菜鱼13 小时前
【Redis】常用数据结构之Hash篇:从常用命令到使用场景详解
数据结构·数据库·redis·后端·缓存·哈希算法
不良人天码星14 小时前
Redis单线程模型为什么快?
数据库·redis·缓存
爱吃烤鸡翅的酸菜鱼14 小时前
【Redis】常用数据结构之List篇:从常用命令到典型使用场景
数据结构·redis·后端·缓存·list
沐雨风栉16 小时前
自建云音乐服务器:Navidrome+cpolar让无损音乐随身听
运维·服务器·redis·缓存·docker·容器
酷酷的崽79819 小时前
Redis 键(Key)的命令
数据库·redis·缓存