Google协议缓冲区,如何处理多个消息类型?

问题描述:

是否可以获取序列化协议缓冲区消息的类型?Google协议缓冲区,如何处理多个消息类型?

我有这样的例子

option java_outer_classname="ProtoUser"; 

message User { 
    required int32 id = 1; 
    required string name = 2; 
    required string firstname = 3; 
    required string lastname = 4; 
    required string ssn= 5; 
} 

message Address { 
    required int32 id = 1; 
    required string country = 2 [default = "US"];; 
    optional string state = 3; 
    optional string city = 4; 
    optional string street = 5; 
    optional string zip = 6; 
} 

在Java中我有这样的代码

Address addr = ProtoUser.Address.newBuilder().setCity("Weston").setCountry("USA").setId(1).setState("FL").setStreet("123 Lakeshore").setZip("90210") 
      .build(); 

    User user = ProtoUser.User.newBuilder().setId(1).setFirstname("Luis").setLastname("Atencio").setName("luisat").setSsn("555-555-5555").build(); 

if(....){ 
    FileOutputStream output = new FileOutputStream("out1.ser"); 
    user.writeTo(output); 
    output.close(); 
}else{ 
    FileOutputStream output = new FileOutputStream("out1.ser"); 
    addr.writeTo(output); 
    output.close(); 
} 

现在,我能确定,如果该文件包含地址还是用户?处理多种消息类型的常用方法是什么?我如何确定我收到的消息类型?

我们无法确定文件是否包含地址或用户。因为数据中没有编码的类型信息。

要处理多个消息类型,可以使用元数据,如:在HTTP在框架基座流协议

    • 扩展的文件名
    • 接头连接
    • 具体帧头.. 。
  • +0

    好吧,这只是一个例子。我将在稍后通过TCP和Unix套接字发送序列化的字符串。可能是像发送'CommandX;'+ serializedString一个很好的解决方案? – user2071938 2014-12-02 12:01:38

    +0

    是的。在框架头部放置一些元数据来描述下面的有效载荷。 – Dante 2014-12-02 13:05:37

    +4

    @ user2071938可以将所有消息包装在公共容器消息中(全部为“可选”或与新的“oneof”联合构造:https://developers.google.com/protocol-buffers/docs/proto#oneof) - 这与在protobuf级别添加标题基本相同。 – zapl 2014-12-02 13:33:19

    要么你:

    1. 包括信息这是否是一个地址或名称的文件结尾(如果文件只包含一种类型在一个时间)
    2. 明确指定数据包的类型的序列化形式,即增加一个字段required int32 type并从该字段中推断出该类型。 (如果两种类型一次包含在一个文件中)
    3. 设计明确包含此信息的特定外部消息格式并包装协议缓冲区。

    无论哪种方式合适,如果您将它们复用到一个通道上 - 您通过选择相同的文件结尾来完成 - 您必须在收到它们时再次解复用它们。