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
相关推荐
hui函数2 小时前
Flask电影投票系统全解析
后端·python·flask
haogexiaole3 小时前
Redis优缺点
数据库·redis·缓存
在未来等你3 小时前
Redis面试精讲 Day 27:Redis 7.0/8.0新特性深度解析
数据库·redis·缓存·面试
小厂永远得不到的男人3 小时前
基于 Spring Validation 实现全局参数校验异常处理
java·后端·架构
川石课堂软件测试5 小时前
技术干货|使用Prometheus+Grafana监控Tomcat实例详解
redis·功能测试·单元测试·tomcat·测试用例·grafana·prometheus
毅航8 小时前
从原理到实践,讲透 MyBatis 内部池化思想的核心逻辑
后端·面试·mybatis
两张不够花8 小时前
Shell脚本源码安装Redis、MySQL、Mongodb、PostgreSQL(无报错版)
linux·数据库·redis·mysql·mongodb·postgresql·云计算
展信佳_daydayup8 小时前
02 基础篇-OpenHarmony 的编译工具
后端·面试·编译器
Always_Passion8 小时前
二、开发一个简单的MCP Server
后端
用户721522078778 小时前
基于LD_PRELOAD的命令行参数安全混淆技术
后端