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

Protobuf3: How to describe map of repeated string?

Official documentation about map type 说:

map<key_type, value_type> map_field = N;

...where the key_type can be any integral or string type (so, any scalar type except for floating point types and bytes). The value_type can be any type.

我想定义一个 map<string, repeated string> 字段,但在我的 libprotoc 3.0.0 上似乎是非法的,它抱怨 Expected ">"。所以我想知道有没有办法把重复的字符串放到map中。

可能的解决方法是:

message ListOfString {
    repeated string value = 1;
}

// Then define:
map<string, ListOfString> mapToRepeatedString = 1;

但是ListOfString这里看起来多余。

我有同样的需求,得到了同样的错误。我不相信这是可能的。这是语言规范中的相关 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 "]" ] ";"

"repeated" 关键字只出现在字段定义中。地图定义需要 "type",其中不包括重复关键字。

这意味着有几个选项。

  • 您可以按照您的指示围绕重复值创建一个包装器。
  • 人们定义地图的方式比较老,比较繁琐但是是等价的。这是语言指南中的向后兼容示例。 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 这是来自其众所周知类型的无类型列表集合。

我觉得应该是这样的。

message ListOfString {
   repeated string what_ever_name = 1;
}

// Then define:
map<string, ListOfString> what_ever_name = 1;

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