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
相关推荐
南囝coding30 分钟前
现代Unix命令行工具革命:30个必备替代品完整指南
前端·后端
夏之小星星42 分钟前
Springboot结合Vue实现分页功能
vue.js·spring boot·后端
小丁爱养花44 分钟前
Redis - set & zset (常用命令/内部编码/应用场景)
数据库·redis·缓存
唐僧洗头爱飘柔95271 小时前
【SpringCloud(8)】SpringCloud Stream消息驱动;Stream思想;生产者、消费者搭建
后端·spring·spring cloud·设计思想·stream消息驱动·重复消费问题
韩立学长1 小时前
【开题答辩实录分享】以《自动售货机刷脸支付系统的设计与实现》为例进行答辩实录分享
vue.js·spring boot·后端
cj6341181502 小时前
DBeaver连接本地MySQL、创建数据库表的基础操作
java·后端
大G的笔记本2 小时前
用 Redis 的 List 存储库存队列,并通过 LPOP 原子性出队来保证并发安全案例
java·数据库·redis·缓存
程序员爱钓鱼3 小时前
Python编程实战—面向对象与进阶语法 | 属性与方法
后端·python·ipython
程序员爱钓鱼3 小时前
Python编程实战——面向对象与进阶语法 | 构造函数与析构函数
后端·python·ipython
逻极3 小时前
Rust之结构体(Structs):构建自定义数据类型
开发语言·后端·rust