0

0

解决UnifiedJedis中JSONGET方法返回字节数组值带.0后缀的问题

花韻仙語

花韻仙語

发布时间:2025-10-23 10:19:47

|

502人浏览过

|

来源于php中文网

原创

解决UnifiedJedis中JSONGET方法返回字节数组值带.0后缀的问题

本文深入探讨了jedis客户端在调用`jsonget`方法时,将json中存储的字节数组值错误地解析为带有`.0`后缀的`double`类型的问题。我们将分析其根源在于底层json库的数值上转型机制,并提供三种实用的解决方案:包括手动类型转换、利用路径指定返回类型以及执行原始redis命令进行数据解析,旨在帮助开发者高效准确地处理redis json数据。

UnifiedJedis jsonGet方法解析字节数组的类型转换问题及解决方案

在使用Jedis客户端(特别是Jedis 4.2.3版本)与Redis进行交互时,开发者可能会遇到一个特定问题:当通过UnifiedJedis.jsonGet方法检索存储在Redis JSON中的字节数组数据时,返回的数值会带有.0的后缀,例如将原始字节值60解析为60.0。这表明Jedis在处理这些数值时,将其上转型为double类型,而非预期的byte或int类型。

问题根源分析

此问题的核心在于Jedis 4.2.3版本内部使用的JSON处理库,如Gson和org.json:json。这些库在默认情况下,会将JSON字符串中的所有数字类型(无论是整数还是浮点数)统一解析为Java的double类型,以确保能够兼容最大范围的数值表示。因此,即使Redis中存储的是表示字节的整数值,经过这些库的解析后,也会被提升为double类型,从而导致输出中出现.0的后缀。

以下是一个典型的示例代码片段,展示了该问题:

import redis.clients.jedis.HostAndPort;
import redis.clients.jedis.UnifiedJedis;

public class JedisJsonGetExample {
    public static void main(String[] args) {
        HostAndPort config = new HostAndPort("localhost", 6379);
        UnifiedJedis client = new UnifiedJedis(config);

        // 假设Redis中键为"StandaloneResponse:..."的JSON数据中,
        // ".variables.ReturnValue"路径下存储的是一个字节数组,例如 [60, 63, 120, ...]
        Object object = client.jsonGet("StandaloneResponse:9b970b5f-32c2-4265-92cb-9af9d6707782");
        System.out.println(object);
        // 输出中,ReturnValue字段会显示为 [60.0, 63.0, 120.0, ...]
        client.close();
    }
}

解决方案

针对这一问题,有多种策略可以确保获取到准确的字节值。以下是三种推荐的方法:

1. 手动类型转换(LinkedHashMap后处理)

这种方法适用于当您不确定具体路径,或者需要处理整个JSON对象,并在获取后进行精细化控制的情况。jsonGet方法通常返回一个Object类型,当数据是JSON对象时,它会被解析为LinkedHashMap。您可以遍历这个Map,识别并转换需要纠正的double值。

实现思路: 将jsonGet返回的Object强制转换为LinkedHashMap,然后递归遍历其内容。当遇到ReturnValue等已知包含字节数组的字段时,将其中的double值转换为byte或int。

示例代码(概念性,需根据实际JSON结构调整):

import redis.clients.jedis.HostAndPort;
import redis.clients.jedis.UnifiedJedis;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.stream.Collectors;

public class JedisJsonGetManualConversion {
    public static void main(String[] args) {
        HostAndPort config = new HostAndPort("localhost", 6379);
        UnifiedJedis client = new UnifiedJedis(config);

        try {
            Object rawObject = client.jsonGet("StandaloneResponse:9b970b5f-32c2-4265-92cb-9af9d6707782");

            if (rawObject instanceof LinkedHashMap) {
                LinkedHashMap dataMap = (LinkedHashMap) rawObject;
                // 假设ReturnValue在variables字段下
                if (dataMap.containsKey("variables")) {
                    Object variablesObj = dataMap.get("variables");
                    if (variablesObj instanceof LinkedHashMap) {
                        LinkedHashMap variablesMap = (LinkedHashMap) variablesObj;
                        if (variablesMap.containsKey("ReturnValue")) {
                            Object returnValueObj = variablesMap.get("ReturnValue");
                            if (returnValueObj instanceof List) {
                                List rawList = (List) returnValueObj;
                                // 将List转换为List或List
                                List byteList = rawList.stream()
                                        .filter(item -> item instanceof Double)
                                        .map(item -> ((Double) item).byteValue())
                                        .collect(Collectors.toList());
                                variablesMap.put("ReturnValue", byteList); // 更新Map中的值
                                System.out.println("Converted ReturnValue: " + byteList);
                            }
                        }
                    }
                }
                System.out.println("Processed Object: " + dataMap);
            } else {
                System.out.println("Raw Object: " + rawObject);
            }
        } finally {
            client.close();
        }
    }
}

注意事项:

  • 这种方法需要您对JSON的结构有清晰的了解。
  • 手动遍历和转换可能增加代码的复杂性,尤其对于深层嵌套的JSON。

2. 利用路径指定返回类型

Jedis的jsonGet方法允许通过Path对象指定要检索的JSON路径,并且可以提供一个目标类来尝试直接反序列化。这是最推荐的方法,因为它简洁高效,并且能够利用Jedis的内置反序列化能力。

Uni-CourseHelper
Uni-CourseHelper

私人AI助教,高效学习工具

下载

实现思路: 使用Path.of()指定到包含字节数组的精确路径,并提供Byte[].class作为期望的返回类型。Jedis会尝试将该路径下的数据直接反序列化为字节数组。

示例代码:

import redis.clients.jedis.HostAndPort;
import redis.clients.jedis.UnifiedJedis;
import redis.clients.jedis.json.Path;

public class JedisJsonGetWithPathAndType {
    public static void main(String[] args) {
        HostAndPort config = new HostAndPort("localhost", 6379);
        UnifiedJedis client = new UnifiedJedis(config);

        try {
            // 假设键为"StandaloneResponse:..."的JSON数据中,
            // 字节数组位于".variables.ReturnValue"路径下
            Byte[] returnValue = client.jsonGet(
                "StandaloneResponse:9b970b5f-32c2-4265-92cb-9af9d6707782",
                Byte[].class,
                Path.of(".variables.ReturnValue")
            );

            if (returnValue != null) {
                System.out.print("Retrieved ReturnValue as Byte array: [");
                for (int i = 0; i < returnValue.length; i++) {
                    System.out.print(returnValue[i]);
                    if (i < returnValue.length - 1) {
                        System.out.print(", ");
                    }
                }
                System.out.println("]");
            } else {
                System.out.println("ReturnValue not found or could not be deserialized.");
            }
        } catch (Exception e) {
            System.err.println("Error retrieving data: " + e.getMessage());
            e.printStackTrace();
        } finally {
            client.close();
        }
    }
}

注意事项:

  • Path的语法必须准确无误,否则可能无法检索到数据。
  • 这种方法依赖于Jedis内部的JSON反序列化器能否正确处理double到byte的转换。在大多数情况下,如果JSON中是整数表示的字节值,Jedis能够正确转换。

3. 执行原始命令并手动解析

如果前两种方法无法满足需求,或者您需要对数据解析过程有极致的控制,可以直接执行Redis的原始JSON命令,并自行解析返回结果。这种方法绕过了Jedis内部的JSON库,直接获取Redis的原始响应。

实现思路: 使用client.executeCommand方法执行JSON.GET命令,该方法返回一个原始的Object,通常是byte[]。然后,您需要根据Redis响应的格式,手动将其转换为字符串,再进行JSON解析。

示例代码:

import redis.clients.jedis.HostAndPort;
import redis.clients.jedis.UnifiedJedis;
import redis.clients.jedis.commands.Protocol.CommandArguments;
import redis.clients.jedis.json.JsonProtocol;
import com.google.gson.Gson; // 或其他JSON库,如Jackson

import java.nio.charset.StandardCharsets;
import java.util.List;
import java.util.Map;

public class JedisJsonGetRawCommand {
    public static void main(String[] args) {
        HostAndPort config = new HostAndPort("localhost", 6379);
        UnifiedJedis client = new UnifiedJedis(config);
        Gson gson = new Gson(); // 用于手动解析JSON

        try {
            // 执行原始JSON.GET命令
            Object rawResult = client.executeCommand(
                new CommandArguments(JsonProtocol.JsonCommand.GET)
                    .add("StandaloneResponse:9b970b5f-32c2-4265-92cb-9af9d6707782")
            );

            if (rawResult instanceof byte[]) {
                String jsonString = new String((byte[]) rawResult, StandardCharsets.UTF_8);
                System.out.println("Raw JSON String: " + jsonString);

                // 手动解析JSON字符串
                // 注意:JSON.GET命令默认返回的是一个JSON数组,即使只查询一个路径
                // 例如:["{\"exchangeId\":...}"]
                List jsonArray = gson.fromJson(jsonString, List.class);
                if (!jsonArray.isEmpty()) {
                    String actualJsonContent = jsonArray.get(0); // 取出实际的JSON对象字符串
                    Map parsedMap = gson.fromJson(actualJsonContent, Map.class);

                    // 从解析后的Map中获取ReturnValue并进行转换
                    if (parsedMap.containsKey("variables")) {
                        Map variables = (Map) parsedMap.get("variables");
                        if (variables != null && variables.containsKey("ReturnValue")) {
                            List rawReturnValue = (List) variables.get("ReturnValue");
                            List byteList = rawReturnValue.stream()
                                    .map(Double::byteValue)
                                    .collect(Collectors.toList());
                            System.out.println("Manually parsed ReturnValue as Byte list: " + byteList);
                        }
                    }
                }
            } else {
                System.out.println("Unexpected raw result type: " + rawResult.getClass().getName());
            }
        } catch (Exception e) {
            System.err.println("Error executing raw command: " + e.getMessage());
            e.printStackTrace();
        } finally {
            client.close();
        }
    }
}

注意事项:

  • executeCommand返回的是Redis协议的原始响应,通常是byte[],需要自行解码为字符串。
  • JSON.GET命令在不指定路径时,返回的是包含整个JSON对象的数组形式的JSON字符串,如["{...}"]。如果指定了路径,则返回该路径下的值数组,例如[60.0, 63.0]。
  • 需要引入一个独立的JSON库(如Gson或Jackson)来解析获取到的JSON字符串。
  • 这种方法提供了最大的灵活性和控制力,但同时也增加了实现的复杂性。

总结与建议

在处理UnifiedJedis.jsonGet方法返回字节数组值带.0后缀的问题时,最佳实践是根据具体需求和对JSON结构的了解程度选择合适的解决方案:

  • 对于已知路径且期望获取特定类型数组的情况,强烈推荐使用“利用路径指定返回类型”的方法。 它最为简洁高效,且能利用Jedis的类型转换能力。
  • 如果需要对整个JSON对象进行处理,或者路径不固定,可以考虑“手动类型转换(LinkedHashMap后处理)”,但需注意其可能带来的代码复杂性。
  • 当需要对数据解析过程有完全控制,或者前两种方法无法满足需求时,才考虑“执行原始命令并手动解析”。这种方法虽然最灵活,但实现成本也最高。

理解Jedis底层JSON库的默认行为是解决此类问题的关键。通过选择合适的策略,可以确保从Redis中准确无误地获取和处理JSON数据。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

418

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

535

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

311

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

77

2025.09.10

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1498

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

623

2023.11.24

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.9万人学习

C# 教程
C# 教程

共94课时 | 7.7万人学习

Java 教程
Java 教程

共578课时 | 52万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号