Language Guide (proto3) | proto3 语言指南(九)Oneof结构
Oneof - Oneof结构
如果消息包含多个字段,并且最多只能同时设置一个字段,则可以使用oneof
功能强制执行此行为并节省内存。
oneof
字段与常规字段类似,但oneof
共享内存中的所有字段除外,并且oneof
最多只能同时设置一个字段。设置oneof
的任何成员将自动清除所有其他成员。您可以使用特殊的case()
或WhichOneof()
方法检查oneof
中设置了哪个值(如果有被设置),具体取决于您选择的语言。
使用oneof
结构
要在.proto
中定义oneof
结构,请使用oneof
关键字后跟oneof名称
。请看示例test_oneof
:
message SampleMessage {
oneof test_oneof {
string name = 4;
SubMessage sub_message = 9;
}
}
然后将oneof
字段添加到oneof
定义。你可以添加任意类型的字段,除了map
和repeated
类型的字段。
在生成的代码中,oneof
字段与常规字段一样具有getter
(访问器)和setter
(设置器)。您还可以使用一种特殊的方法来检查oneof
结构中设置了哪个值(如果有的话)。您可以在相关的API参考中找到有关所选语言的oneof API
的更多信息。
oneof
的功能
- 设置oneof字段将自动清除oneof的所有其他成员。因此,如果您设置了几个字段中的一个,那么只有最后设置的字段有效。
SampleMessage message;
message.set_name("name");
CHECK(message.has_name());
message.mutable_sub_message(); // Will clear name field.
CHECK(!message.has_name());
- 如果解析器在
wire
上遇到同一个oneof
的多个成员,则在解析的消息中只使用看到的最后一个成员。 oneof
结构不能使用repeated
修饰符。- 反射API适用于
oneof
字段。 - 如果将
oneof
字段设置为默认值(例如将int32类型的oneof
字段设置为0),则将设置该oneof
字段的case
,并且该值将在wire
上序列化。 - 如果使用
C++
,请确保代码不会导致内存崩溃。下面的示例代码将崩溃,因为在调用set_name()
方法时已经删除了sub_message
。
SampleMessage message;
SubMessage* sub_message = message.mutable_sub_message();
message.set_name("name"); // Will delete sub_message
sub_message->set_... // Crashes here
- 同样在
C++
中,如果你使用Swap()
方法交换两条使用了oneof
结构的消息,每条消息都将以另外一条消息的oneof case
结束:在下面的示例中,msg1
将有一个sub_message
字段,msg2
将有一个name
字段。
SampleMessage msg1;
msg1.set_name("name");
SampleMessage msg2;
msg2.mutable_sub_message();
msg1.swap(&msg2);
CHECK(msg1.has_sub_message());
CHECK(msg2.has_name());
向后(下)兼容问题
添加或删除oneof
字段时要小心。如果检查oneof
字段的值返回None/NOT_SET
,则可能意味着此oneof
尚未设置或已将其设置为oneof
的其他版本中的字段。因为无法知道wire
的未知字段是否是其中一个字段的成员,所以无法区分两者之间的区别。
标签重用问题
- 将字段移入或移出
oneof
:消息序列化和解析后,可能会丢失一些信息(某些字段将被清除)。但是,您可以安全地将single
字段移动到新的oneof
字段中,并且如果已知只设置了一个字段,则可以移动多个字段。 - 删除
oneof
字段并将其加回:这可能会在序列化和解析消息后清除当前设置的oneof
字段。 - 拆分或合并
oneof
字段:这与移动常规字段有类似的问题
Language Guide (proto3) | proto3 语言指南(九)Oneof结构的更多相关文章
- Swift语言指南(九)--基本运算符
原文:Swift语言指南(九)--基本运算符 运算符(operator)是用来检查,改变或合并值的一种特殊符号或短语.例如,加号运算符让两个数字相加(如:let i = 1 + 2),还有些更复杂的运 ...
- Swift5 语言指南(九) 闭包
闭包是自包含的功能块,可以在代码中传递和使用.Swift中的闭包类似于C和Objective-C中的块以及其他编程语言中的lambdas. 闭包可以从定义它们的上下文中捕获和存储对任何常量和变量的引用 ...
- Language Guide (proto3) | proto3 语言指南(开篇)
前言 近日在学习gRPC框架的相关知识时接触到Protobuf(protocol-buffers,协议缓冲区),proto3等知识.网上很多文章/帖子经常把gRPC与proto3放在一起,为避免初学者 ...
- Language Guide (proto3) | proto3 语言指南(十五)生成类
Generating Your Classes - 生成类 要生成Java.Python.C++.Go.Ruby.ObjuleC或C代码,需要使用.proto文件中定义的消息类型,还需要在.proto ...
- Language Guide (proto3) | proto3 语言指南(十四)选项
Options - 选项 .proto文件中的单个声明可以使用许多 选项 进行注释.选项不会更改声明的总体含义,但可能会影响在特定上下文中处理声明的方式.可用选项的完整列表在google/protob ...
- Language Guide (proto3) | proto3 语言指南(十二)定义服务
Defining Services - 定义服务 如果要在RPC(Remote Procedure Call,远程过程调用)系统中使用消息类型,可以在.proto文件中定义RPC服务接口,协议缓冲区编 ...
- Protobuf 语言指南(proto3)
Protobuf 语言指南(proto3) Protocol Buffer是Google的语言中立的,平台中立的,可扩展机制的,用于序列化结构化数据 - 对比XML,但更小,更快,更简单.您可以定义数 ...
- Protocol Buffers(Protobuf) 官方文档--Protobuf语言指南
Protocol Buffers(Protobuf) 官方文档--Protobuf语言指南 约定:为方便书写,ProtocolBuffers在下文中将已Protobuf代替. 本指南将向您描述如何使用 ...
- Protobuf语言指南(转)
Protobuf语言指南 l 定义一个消息(message)类型 l 标量值类型 l Optional 的字段及默认值 l 枚举 l 使用其他消息类型 l 嵌套类型 l 更新一个消息类型 ...
随机推荐
- 数据接口请求异常:parsererror
问题一:直接拿别人的文件放在本地打开 如下图 原因:这是提示"交叉源请求仅支持协议方案:HTTP.数据.Chrome.Chrome扩展.HTTPS." 也就是你不能用本地文件打开, ...
- Nginx压力测试问题
[root@aa~]# This is ApacheBench, Version 2.3 <Revision:655654Revision:655654> Copyright 1996 A ...
- 解决threadLocal父子变量传递问题
一.问题的提出 在系统开发过程中常使用ThreadLocal进行传递日志的RequestId,由此来获取整条请求链路.然而当线程中开启了其他的线程,此时ThreadLocal里面的数据将会出现无法获取 ...
- noip2016自测报告
随便测了一下2016的提高组,发现自己还是太菜了... day1 T1:玩具谜题 一个模拟题,打挂了,只有95分,错的很傻逼 code: #include<bits/stdc++.h> u ...
- Maven的安装跟配置(最全)
一.去官网下载maven 官网地址: https://maven.apache.org/ 点击下载apache-maven-3.6.3-bin.zip 下载完成后解压即可. 二.配置环境变量 在我们的 ...
- C#中的格式
格式模式 说明 : d 月中的某一天.一位数的日期没有前导零. dd 月中的某一天.一位数的日期有一个前导零. ddd 周中某天的缩写名称,在 AbbreviatedDayNames 中定义. ddd ...
- <input type="image">表单提交2次 重复插入数据问题
写一个表单提交用到图片:两种代码. <input type="image" src="xxx.gif"onclick="return dosub ...
- Spark学习进度-Transformation算子
Transformation算子 intersection 交集 /* 交集 */ @Test def intersection(): Unit ={ val rdd1=sc.parallelize( ...
- NOIP初赛篇——02计算机系统的基本结构
引言 计算机系统由硬件和软件两部分组成,硬件系统是计算机的"躯干",是物质基础.而软件系统则是建立在这个"躯干"上的"灵魂". 计算机硬件 ...
- Dota游戏匹配的所有组合
在Dota游戏中有一种匹配玩法,任意5人以下玩家组队,加入匹配系统,由系统组合出5人 vs 5人的组合进行游戏,比如2人+3人 vs 1人+4人.抽象出这个问题,就变成两边各有m个玩家,最多允许n个 ...