redis 反序列化 isSuccess()序列化失败问题

前言

公司统一封装了返回bash对象R<T> 其中有个方法 isCuccess 来判断接口是否成功。由于性能优化把部分接口返回的对象给缓存了 R<Object>存储在redis 序列化成功,但是反序列化失败了。我们来看下具体问题

定位问题

R<T> 对象 重点在 isSuccess 方法

scss 复制代码
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Accessors(chain = true)
public class R<T> implements Serializable {

    /**
     * 成功标记
     */
    public static final Integer SUCCESS = 200;
    /**
     * 失败标记
     */
    public static final Integer FAIL = -1;
    /**
     *
     */
    private static final long serialVersionUID = 1L;
    /**
     * code
     */
    private Integer code;
    /**
     * msg
     */
    private String msg;

    /**
     * data
     */
    private T data;

    /**
     * currentTimeMills
     */
    @Builder.Default
    private Long currentTimeMillis = System.currentTimeMillis();

    public static <T> R<T> ok() {
        return restResult(null, SUCCESS, null);
    }

    public static <T> R<T> ok(T data) {
        return restResult(data, SUCCESS, null);
    }

    public static <T> R<T> ok(T data, String msg) {
        return restResult(data, SUCCESS, msg);
    }

    public static <T> R<T> failed() {
        return restResult(null, FAIL, null);
    }

    public static <T> R<T> failed(String msg) {
        return restResult(null, FAIL, msg);
    }

    public static <T> R<T> failed(T data) {
        return restResult(data, FAIL, null);
    }

    public static <T> R<T> failed(T data, String msg) {
        return restResult(data, FAIL, msg);
    }

    public static <T> R<T> restResult(T data, Integer code, String msg) {
        return R.<T>builder().code(code).data(data).msg(msg).build();
    }

    public static R<Void> failure(Integer code, String message) {
        return R.<Void>builder().code(code).msg(message).build();
    }

    /**
     * @return boolean
     */
    public boolean isSuccess() {
        return SUCCESS.equals(code);
    }
}

redis 存储的数据

json 复制代码
[
  "com.wf.cabbage.web.base.R",
  {
    "code": 200,
    "msg": null,
    "data": [
      "java.util.ArrayList",
      []
    ],
    "currentTimeMillis": 1712662374888,
    "success": true
  }
]

错误

text 复制代码
org.springframework.data.redis.serializer.SerializationException: Could not read JSON: Unrecognized field "success" (class com.xx.cabbage.web.base.R), not marked as ignorable (4 known properties: "msg", "data", "currentTimeMillis", "code"])
 at [Source: (byte[])"["com.xx.cabbage.web.base.R",{"code":200,"msg":null,"data":["java.util.ArrayList",[]],"currentTimeMillis":1712662374888,"success":true}]"; line: 1, column: 135] (through reference chain: com.xx.cabbage.web.base.R["success"]); nested exception is com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException: Unrecognized field "success" (class com.xx.cabbage.web.base.R), not marked as ignorable (4 known properties: "msg", "data", "currentTimeMillis", "code"])
 at [Source: (byte[])"["com.xx.cabbage.web.base.R",{"code":200,"msg":null,"data":["java.util.ArrayList",[]],"currentTimeMillis":1712662374888,"success":true}]"; line: 1, column: 135] (through reference chain: com.xx.cabbage.web.base.R["success"])

解决

这是因为 redis 我们这边使用的序列化是 GenericJackson2JsonRedisSerializer 使用的是 Jackson 库进行序列化和反序列化,而 Jackson 库默认会将 Java 对象中的 getter 方法转换为 JSON 对象中的属性。因此,在反序列化 Redis 中的 JSON 数据时,Jackson 库会将 isSuccess() 方法转换为 JSON 对象中的 success 属性。

要解决这个问题,你可以采用以下两种方式之一:

  1. 在 R 类中添加一个名为 "success" 的 getter 方法,并在该方法中调用 isSuccess() 方法。这样,在反序列化 Redis 中的 JSON 数据时,Jackson 库就可以正确地将 success 属性映射到 Java 对象的 isSuccess() 方法中。
  2. 修改 GenericJackson2JsonRedisSerializer 的配置,禁用 Jackson 库的 getter 方法转换功能。具体来说,你可以通过以下代码来创建 GenericJackson2JsonRedisSerializer 对象: 核心配置 objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.NONE); objectMapper.setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY);
ini 复制代码
ObjectMapper objectMapper = new ObjectMapper();

objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.NONE);
objectMapper.setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY);
objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
// 支持 java 1.8+ LocalDateTime 序列化
objectMapper.registerModule(new JavaTimeModule());
objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);

GenericJackson2JsonRedisSerializer genericJackson2JsonRedisSerializer =
       new GenericJackson2JsonRedisSerializer(objectMapper);

this.setDefaultSerializer(genericJackson2JsonRedisSerializer);

this.setKeySerializer(new StringRedisSerializer());

this.setHashKeySerializer(new StringRedisSerializer());
this.setHashValueSerializer(genericJackson2JsonRedisSerializer);

this.setValueSerializer(genericJackson2JsonRedisSerializer);
相关推荐
MY_TEUCK5 分钟前
Sealos 平台部署实战指南:结合 Cursor 与版本发布流程
java·人工智能·学习·aigc
我爱cope35 分钟前
【从0开始学设计模式-10| 装饰模式】
java·开发语言·设计模式
朝新_1 小时前
【Spring AI 】图像与语音模型实战
java·人工智能·spring
小码哥_常1 小时前
告别臃肿!Elasticsearch平替Manticore登场
后端
RH2312112 小时前
2026.4.16Linux 管道
java·linux·服务器
zmsofts2 小时前
java面试必问13:MyBatis 一级缓存、二级缓存:从原理到脏数据,一篇讲透
java·面试·mybatis
苍何2 小时前
万字保姆级教程:Hermes+Kimi K2.6 打造7x24h Agent军团
后端
我叫黑大帅2 小时前
为什么map查找时间复杂度是O(1)?
后端·算法·面试
aq55356003 小时前
编程语言三巨头:汇编、C++与PHP大比拼
java·开发语言
我是无敌小恐龙4 小时前
Java SE 零基础入门Day01 超详细笔记(开发前言+环境搭建+基础语法)
java·开发语言·人工智能·opencv·spring·机器学习