Protobuf3:如何描述重复字符串的映射?

lz9*_*z96 11 protocol-buffers protocol-buffers-3

有关地图类型官方文档说:

map<key_type, value_type> map_field = N;

...其中key_type可以是任何整数或字符串类型(因此,除了浮点类型和字节之外的任何标量类型).value_type可以是任何类型.

我想定义一个map<string, repeated string>字段,但它对我来说似乎是非法的libprotoc 3.0.0,它抱怨Expected ">".所以我想知道是否有任何方法可以将重复的字符串放入地图中.

可能的解决方法可能是:

message ListOfString {
    repeated string value = 1;
}

// Then define:
map<string, ListOfString> mapToRepeatedString = 1;
Run Code Online (Sandbox Code Playgroud)

但这ListOfString看起来多余.

小智 10

我有同样的需求,并得到了同样的错误.我不相信这是可能的.以下是语言规范中的相关BNF定义.

https://developers.google.com/protocol-buffers/docs/reference/proto3-spec

messageType = [ "." ] { ident "." } messageName
mapField = "map" "<" keyType "," type ">" mapName "=" fieldNumber [ "["fieldOptions "]" ] ";"
type = "double" | "float" | "int32" | "int64" | "uint32" | "uint64"
  | "sint32" | "sint64" | "fixed32" | "fixed64" | "sfixed32" | "sfixed64"
  | "bool" | "string" | "bytes" | messageType | enumType
messageName = ident
ident = letter { letter | decimalDigit | "_" }
field = [ "repeated" ] type fieldName "=" fieldNumber [ "[" fieldOptions "]" ] ";"
Run Code Online (Sandbox Code Playgroud)

"重复"关键字仅出现在字段定义中.地图定义需要"类型",不包括重复的关键字.

这意味着有一些选择.

  • 您可以按照指示在重复值周围创建一个包装器.
  • 人们定义了较旧的定义地图的方式,这更加繁琐,但却是等价的.这是语言指南中的向后兼容示例.https://developers.google.com/protocol-buffers/docs/proto3#maps
        message MapFieldEntry {
          key_type key = 1;
          repeated value_type value = 2;
        }
        repeated MapFieldEntry map_field = N;
        
    您需要自己将数据转换为地图,但在大多数语言中这应该是相当简单的.在Java中:
        List<MapFieldEntry> map_field = // Existing list from protobuf.
        Map<key_type, List<value_type>> = map_field.stream()
            .collect(Collectors.toMap(kv -> kv.key, kv -> kv.value));
    
        
  • 使用google.protobuf.ListValue https://developers.google.com/protocol-buffers/docs/reference/google.protobuf#listvalue这是一个来自众所周知类型的无类型列表集合.

  • 我很好奇为什么 Protobuf 同时具有“ListValue”和“repeated”。它们的目的不一样吗?关于“ListValue”的直接文档有些稀疏。 (3认同)

Vir*_*ths 6

我认为应该如下。

message ListOfString {
   repeated string what_ever_name = 1;
}

// Then define:
map<string, ListOfString> what_ever_name = 1;
Run Code Online (Sandbox Code Playgroud)

请记住what_ever_name在两个地方都应该相同。

  • 为什么两个地方的“what_ever_name”字段名称需要相同?这不是一个完全独立的类型吗? (3认同)
  • @BigMoney 不一定是这样。我认为这个例子就是这样阐述的。 (3认同)