跳到主要内容
版本:1.0.0

自定义序列化器

自定义序列化器允许你精确定义某个类型如何序列化和反序列化。这适用于需要特殊处理、优化或跨语言兼容性的类型。

何时使用自定义序列化器

  • 特殊编码:需要特定二进制格式的类型
  • 第三方类型:来自外部库、Fory 无法自动处理的类型
  • 优化:当你能比默认的基于反射的方法更高效地序列化时
  • 跨语言兼容性:当互操作需要精确控制二进制格式时

ExtensionSerializer 接口

自定义序列化器实现 ExtensionSerializer 接口:

type ExtensionSerializer interface {
// WriteData 将值序列化到缓冲区。
// 只写入数据,Fory 会处理类型信息和引用。
// 使用 ctx.Buffer() 访问 ByteBuffer。
// 使用 ctx.SetError() 上报错误。
WriteData(ctx *WriteContext, value reflect.Value)

// ReadData 从缓冲区反序列化值,并写入提供的 value。
// 只读取数据,Fory 会处理类型信息和引用。
// 使用 ctx.Buffer() 访问 ByteBuffer。
// 使用 ctx.SetError() 上报错误。
ReadData(ctx *ReadContext, value reflect.Value)
}

基础示例

下面是一个带整数号字段的类型所使用的简单自定义序列化器:

import (
"reflect"
"github.com/apache/fory/go/fory"
)

type MyExt struct {
Id int32
}

type MyExtSerializer struct{}

func (s *MyExtSerializer) WriteData(ctx *fory.WriteContext, value reflect.Value) {
myExt := value.Interface().(MyExt)
ctx.Buffer().WriteVarint32(myExt.Id)
}

func (s *MyExtSerializer) ReadData(ctx *fory.ReadContext, value reflect.Value) {
id := ctx.Buffer().ReadVarint32(ctx.Err())
value.Set(reflect.ValueOf(MyExt{Id: id}))
}

// 注册自定义序列化器
f := fory.New(fory.WithXlang(true))
err := f.RegisterExtension(MyExt{}, 100, &MyExtSerializer{})

Context 方法

WriteContextReadContext 提供对序列化资源的访问:

方法说明
Buffer()返回用于读写的 *ByteBuffer
Err()返回 *Error,用于延迟错误检查
SetError(err)在 context 上设置错误
HasError()如果已设置错误,则返回 true
TypeResolver()返回用于嵌套类型的类型解析器
RefResolver()返回用于引用支持的引用解析器

ByteBuffer 方法

ByteBuffer 提供读写基本类型的方法:

写入方法

方法说明
WriteBool(v bool)写入布尔值
WriteInt8(v int8)写入有符号 8 位整数
WriteInt16(v int16)写入有符号 16 位整数
WriteInt32(v int32)写入有符号 32 位整数
WriteInt64(v int64)写入有符号 64 位整数
WriteFloat32(v float32)写入 32 位浮点数
WriteFloat64(v float64)写入 64 位浮点数
WriteVarint32(v int32)写入变长有符号 32 位整数
WriteVarint64(v int64)写入变长有符号 64 位整数
WriteBinary(data []byte)写入原始字节

读取方法

所有读取方法都接受一个 *Error 参数,用于延迟错误检查:

方法说明
ReadBool(err *Error) bool读取布尔值
ReadInt8(err *Error) int8读取有符号 8 位整数
ReadInt16(err *Error) int16读取有符号 16 位整数
ReadInt32(err *Error) int32读取有符号 32 位整数
ReadInt64(err *Error) int64读取有符号 64 位整数
ReadFloat32(err *Error) float32读取 32 位浮点数
ReadFloat64(err *Error) float64读取 64 位浮点数
ReadVarint32(err *Error) int32读取变长有符号 32 位整数
ReadVarint64(err *Error) int64读取变长有符号 64 位整数
ReadBinary(length int, err *Error) []byte读取指定长度的原始字节

复杂类型示例

下面是一个包含多个字段的类型的自定义序列化器:

type Point3D struct {
X, Y, Z float64
Label string
}

type Point3DSerializer struct{}

func (s *Point3DSerializer) WriteData(ctx *fory.WriteContext, value reflect.Value) {
p := value.Interface().(Point3D)
buf := ctx.Buffer()
buf.WriteFloat64(p.X)
buf.WriteFloat64(p.Y)
buf.WriteFloat64(p.Z)
// 将字符串写为长度 + 字节
labelBytes := []byte(p.Label)
buf.WriteVarint32(int32(len(labelBytes)))
buf.WriteBinary(labelBytes)
}

func (s *Point3DSerializer) ReadData(ctx *fory.ReadContext, value reflect.Value) {
buf := ctx.Buffer()
err := ctx.Err()
x := buf.ReadFloat64(err)
y := buf.ReadFloat64(err)
z := buf.ReadFloat64(err)
labelLen := buf.ReadVarint32(err)
labelBytes := buf.ReadBinary(int(labelLen), err)
value.Set(reflect.ValueOf(Point3D{
X: x,
Y: y,
Z: z,
Label: string(labelBytes),
}))
}

f := fory.New(fory.WithXlang(true))
f.RegisterExtension(Point3D{}, 101, &Point3DSerializer{})

处理指针

当类型包含指针时,需要显式处理 nil 值:

type OptionalValue struct {
Value *int64
}

type OptionalValueSerializer struct{}

func (s *OptionalValueSerializer) WriteData(ctx *fory.WriteContext, value reflect.Value) {
ov := value.Interface().(OptionalValue)
buf := ctx.Buffer()
if ov.Value == nil {
buf.WriteBool(false) // nil 标志
} else {
buf.WriteBool(true) // 非 nil
buf.WriteInt64(*ov.Value)
}
}

func (s *OptionalValueSerializer) ReadData(ctx *fory.ReadContext, value reflect.Value) {
buf := ctx.Buffer()
err := ctx.Err()
hasValue := buf.ReadBool(err)
if !hasValue {
value.Set(reflect.ValueOf(OptionalValue{Value: nil}))
return
}
v := buf.ReadInt64(err)
value.Set(reflect.ValueOf(OptionalValue{Value: &v}))
}

错误处理

使用 ctx.SetError() 上报错误:

func (s *MySerializer) ReadData(ctx *fory.ReadContext, value reflect.Value) {
buf := ctx.Buffer()
version := buf.ReadInt8(ctx.Err())
if ctx.HasError() {
return
}

if version != 1 {
ctx.SetError(fory.DeserializationErrorf("unsupported version: %d", version))
return
}
// 继续读取...
value.Set(reflect.ValueOf(result))
}

注册选项

按 ID 注册

序列化更紧凑,但要求跨语言协调 ID:

f.RegisterExtension(MyType{}, 100, &MySerializer{})

按名称注册

更灵活,但序列化成本更高,因为序列化数据会包含类型名:

f.RegisterExtensionByName(MyType{}, "myapp.MyType", &MySerializer{})

最佳实践

  1. 保持简单:只序列化真正需要的内容
  2. 使用变长整数:对经常较小的整数使用 WriteVarint32/WriteVarint64
  3. 显式处理 nil:检查 nil 指针和 slice
  4. 为格式设置版本:考虑添加版本字节,为未来兼容性预留空间
  5. 测试往返:始终验证 Read(Write(value)) == value
  6. 匹配读写顺序:读取字段的顺序必须与写入顺序完全一致
  7. 检查错误:读取后使用 ctx.HasError() 优雅地处理错误
  8. 先部署再使用:发送由注册序列化器生成的数据前,务必先把该序列化器部署到所有服务。如果服务收到未注册序列化器对应的数据,反序列化会失败

测试自定义序列化器

func TestMySerializer(t *testing.T) {
f := fory.New(fory.WithXlang(true))
f.RegisterExtension(MyType{}, 100, &MySerializer{})

original := MyType{Field: "test"}

// 序列化
data, err := f.Serialize(original)
require.NoError(t, err)

// 反序列化
var result MyType
err = f.Deserialize(data, &result)
require.NoError(t, err)

assert.Equal(t, original, result)
}

相关主题