31

关于地图类型的官方文档说:

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;

ListOfString这里看起来是多余的。

4

2 回答 2

19

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

“重复”关键字仅出现在字段定义中。地图定义需要一个“类型”,它不包括重复的关键字。

这意味着有几个选项。

  • 您可以按照您的指示围绕重复值创建一个包装器。
  • 人们定义地图的旧方式是定义地图,这种方式更加繁琐但等效。这是语言指南中向后兼容的示例。https://developers.google.com/protocol-buffers/docs/proto3#maps
        消息 MapFieldEntry {
          键类型键 = 1;
          重复 value_type value = 2;
        }
        重复 MapFieldEntry map_field = N;
        
    您需要自己将数据转换为地图,但这在大多数语言中应该是相当简单的。在 Java 中:
        List<MapFieldEntry> map_field = // 来自 protobuf 的现有列表。
        地图<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 这是他们众所周知的类型的无类型列表集合。
于 2017-01-09T16:54:38.573 回答
8

我认为应该如下。

message ListOfString {
   repeated string what_ever_name = 1;
}

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

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

于 2019-09-23T01:01:01.090 回答