Redisson 自定义序列化 Codec 实战指南:原理、实现与踩坑记录

Redisson 自定义序列化 Codec 实战指南:原理、实现与踩坑记录

📚 背景介绍:为什么要自定义 Codec?

Redisson 默认支持多种序列化方式(如 JsonJacksonCodec, FstCodec, KryoCodec 等),但在实际项目中可能遇到如下问题:

  • 默认的序列化器无法满足精度、兼容性或安全性要求
  • 要与其他服务(如 Python/Golang)兼容通信格式
  • 想要替换掉过大的 Jackson,换成轻量的 JSON 序列化方式
  • 需要对某些字段做脱敏或加密处理

🧩 Codec 接口原理简析

java 复制代码
public interface Codec extends Serializable {
    Decoder<Object> getValueDecoder();
    Encoder getValueEncoder();
    Decoder<Object> getMapValueDecoder();
    Encoder getMapValueEncoder();
    Decoder<Object> getMapKeyDecoder();
    Encoder getMapKeyEncoder();
}

✍️ 实战演练:自定义 FastJson Codec

java 复制代码
public class FastJsonCodec implements Codec {
    private final Charset charset = StandardCharsets.UTF_8;

    @Override public Decoder<Object> getValueDecoder() { return this::decode; }
    @Override public Encoder getValueEncoder() { return this::encode; }
    @Override public Decoder<Object> getMapValueDecoder() { return getValueDecoder(); }
    @Override public Encoder getMapValueEncoder() { return getValueEncoder(); }
    @Override public Decoder<Object> getMapKeyDecoder() { return getValueDecoder(); }
    @Override public Encoder getMapKeyEncoder() { return getValueEncoder(); }

    private byte[] encode(Object in) throws IOException {
        if (in == null) return new byte[0];
        String json = JSON.toJSONString(in);
        return json.getBytes(charset);
    }

    private Object decode(ByteBuffer buf, ClassLoader classLoader) throws IOException {
        if (!buf.hasRemaining()) return null;
        byte[] bytes = new byte[buf.remaining()];
        buf.get(bytes);
        return JSON.parse(new String(bytes, charset));
    }
}

🔧 配置与使用

java 复制代码
Config config = new Config();
config.useSingleServer().setAddress("redis://127.0.0.1:6379");
config.setCodec(new FastJsonCodec());
RedissonClient redisson = Redisson.create(config);

🧪 测试验证

java 复制代码
@Test
public void testFastJsonCodec() {
    RMap<String, MyDTO> map = redisson.getMap("myMap");
    map.put("key", new MyDTO("test", 123));
    MyDTO value = map.get("key");

    assertEquals("test", value.getName());
    assertEquals(123, value.getValue());
}

⚠️ 踩坑与注意事项

  1. 类型信息丢失:使用 WriteClassName 写入类型,启用 autoType 支持(有安全风险)
  2. 编码一致性:统一使用 UTF-8 避免乱码
  3. 服务兼容性:跨语言建议使用 JSON 或 Protobuf
相关推荐
用户409932250212几秒前
如何用 Git Hook 和 CI 流水线为 FastAPI 项目保驾护航?
后端·ai编程·trae
召摇4 分钟前
Java 21到25的核心API演进总结
java·后端
buddy_red11 分钟前
Knox工具调用功能测试
人工智能·后端·程序员
知其然亦知其所以然15 分钟前
SpringAI 玩转 OCI GenAI:这次我们聊聊 Cohere 聊天模型
java·后端·spring
种子q_q19 分钟前
Redis的三种典型的 “缓存失效” 问题
后端·面试
金銀銅鐵20 分钟前
[Java] 观察 CompactStrings 选项的影响
java·后端
程序猿二饭20 分钟前
Spring Boot 项目启动报错:MongoSocketOpenException 连接被拒绝排查日记
后端
UP21 分钟前
【C++基础】内存管理——malloc/free和new/delete之间的盘根错节
后端
齐穗穗24 分钟前
springboot集成websocket
spring boot·后端·websocket
玉衡子27 分钟前
四、索引优化实战
java·后端