protobuf自解释message

本文介绍了一种通过将proto定义和序列化的数据组合成一个自解释对象的方法,实现了无需客户端提前了解proto定义即可解析消息的功能。这种方法适用于需要高度灵活的数据结构场景。

将proto的定义和序列化的数据组成一个对象,在解码时使用message内部存储的proto定义和数据就可以实现proto消息的自解释。

代码

在proto发布的包内自带了descriptor引入该类型组装成如下格式:

syntax = "proto2";
import 'google/protobuf/descriptor.proto';

message SelfDescribingMessage {
  // proto的定义
  required google.protobuf.FileDescriptorProto proto_file = 1;

  // 具体的message类型,由于一个proto中可以包含多个message
  required string type_name = 2;

  // 具体序列化的数据
  required bytes message_data = 3;
}

然后可以随便定义一个proto类型

syntax = "proto2";
message Person{
    optional string name=1;
    optional int32 age=2;
}

下面进行自定义类型的序列化和反序列化:

        //序列化
        //创建对象
        byte[] data=Type.Person.newBuilder().setName("Myname").setAge(18).build().toByteArray();
        //获得proto定义
        DescriptorProtos.FileDescriptorProto desc=Type.getDescriptor().toProto();
        
        //组件自解释对象,typename是具体message的名字
        Demo.SelfDescribingMessage message=Demo.SelfDescribingMessage.newBuilder().setProtoFile(desc)
                .setTypeName("Person").setMessageData(ByteString.copyFrom(data)).build();
        //进行序列化
        FileOutputStream fos=new FileOutputStream("D://message");
        message.writeTo(fos);
        fos.close();

        //反序列化
        //读取文件
        FileInputStream fis=new FileInputStream("D://message");
        Demo.SelfDescribingMessage fileMessage=Demo.SelfDescribingMessage.parseFrom(fis);
        //获得proto文件
        DescriptorProtos.FileDescriptorProto fdp=fileMessage.getProtoFile();
        Descriptors.FileDescriptor[] fds={};
        //根据typename获得具体的message定义
        Descriptors.Descriptor fileDesc=Descriptors.FileDescriptor.buildFrom(fdp,fds).findMessageTypeByName(fileMessage.getTypeName());
        //读出消息并打印
        System.out.println(DynamicMessage.parseFrom(fileDesc,fileMessage.getMessageData().toByteArray()));
        fis.close();

下面就是程序的输出

name: "Myname"
age: 18

总结

通过如上方法可以不用事先将proto的定义让客户端知道,而是将定义随着消息一起打包,对于需要极度灵活结构的需求可以使用。

转载于:https://www.cnblogs.com/resentment/p/6617803.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值