java - 如何将字符串反序列化为 Java 对象

标签 java utf-8 redis deserialization spring-data-redis

我尝试从用于存储 Spring session 的 Redis 服务器将字符串反序列化为 Java 对象,并且我想在 Spring 框架之外将其反序列化。我认为 Spring Redis 序列化程序可能使用默认字符集 UTF-8 将 Java 对象序列化为字符串。

Redis 中的字符串:

\xAC\xED\x00\x05sr\x00\x0Ejava.lang.Long;\x8B\xE4\x90\xCC\x8F#\xDF\x02\x00\x01J\x00\x05valuexr\x00\x10java.lang.Number\x86\xAC\x95\x1D\x0B\x94\xE0\x8B\x02\x00\x00xp\x00\x00\x00\x00\x00\x00\xD7

我使用的方法:

private static void scanKeys(Jedis jedis, String prefix, String hashKey)
        throws IOException, ClassNotFoundException {
    Set<String> keys = jedis.keys(prefix + "*");

    for (String key : keys) {
        if (!key.contains("expir")) {
            Map<String, String> sessionMap = jedis.hgetAll(key);

            for (Entry<String, String> entry : sessionMap.entrySet()) {

                if (entry.getValue() != null && entry.getKey().equals(hashKey)) {
                    System.out.println(entry.getKey());
                    System.out.println(entry.getValue());

                    byte[] output = entry.getValue().getBytes(charset);
                    System.out.println(Arrays.toString(output));    
                    try {
                        Long id = (Long) deserialize(output);
                        System.out.println(id);
                    } catch (EOFException e) {
                        break;
                    }
                }
            }

        }

    }
}

public static Object deserialize(byte[] data) throws IOException, ClassNotFoundException {
    ByteArrayInputStream in = new ByteArrayInputStream(data);
    ObjectInputStream is = new ObjectInputStream(in);
    return is.readObject();
}

错误信息:

java.io.StreamCorruptedException: invalid stream header: EFBFBDEF
    at java.io.ObjectInputStream.readStreamHeader(ObjectInputStream.java:862)
    at java.io.ObjectInputStream.<init>(ObjectInputStream.java:354)
    at practice.redis.jedis.App.deserialize(App.java:191)
    at practice.redis.jedis.App.scanKeys(App.java:168)
    at practice.redis.jedis.App.main(App.java:77)

我知道 byte[] 和 String 之间 UTF-8 的转换可能是问题所在,但我还是想问问有没有人知道如何在不修改序列化部分的情况下解决这个问题。

最佳答案

我建议您自己解析字符串。它看起来像是一个简单的转义和非转义字节序列。

String value = "\\xAC\\xED\\x00\\x05sr\\x00\\x0Ejava.lang.Long;\\x8B\\xE4\\x90\\xCC\\x8F#\\xDF\\x02\\x00\\x01J\\x00\\x05valuexr\\x00\\x10java.lang.Number\\x86\\xAC\\x95\\x1D\\x0B\\x94\\xE0\\x8B\\x02\\x00\\x00xp\\x00\\x00\\x00\\x00\\x00\\x00\\xD7";

try (ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream()) {
    int i = 0;
    while (i < value.length()) {
        char ch = value.charAt(i);
        if (ch == '\\') {
            if (value.charAt(i + 1) != 'x') {
                throw new UnsupportedOperationException();
            }
            String hex = value.substring(i + 2, i + 4);
            byteArrayOutputStream.write(Integer.parseInt(hex, 16) & 0xFF);
            i += 4;
        } else {
            byteArrayOutputStream.write(ch);
            i++;
        }
    }

    byte[] output = byteArrayOutputStream.toByteArray();
    Long id = (Long) deserialize(output);
    System.out.println(id);
}

关于java - 如何将字符串反序列化为 Java 对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50473724/

相关文章:

python - 如何正确解码以十进制书写的字符串中的转义序列

java - 使用 RandomAccessFile 读取单个 UTF-8 字符

spring-boot - 如何在 Spring Boot 2.0 中使用 Redis 作为数据库映射两个实体之间的关系?

performance - nodejs http 和 redis,只有 6000req/s

java - 链表的 'head'是什么?

java - 使用正则表达式或 xpath 从包含特定文本的 html 中提取 <td> 值

java - 从字符串末尾删除无关字符的正则表达式是什么?

java - 如何找到与特定值最接近的数组元素之和?

c# - 如何删除 4 字节字符?

javascript - 需要外部输入的 Node.js 单例模块模式