在定义protobuf消息时,有时候需要用到extensions来对原有的消息类型进行扩展,有利于消息定义的重复使用。
1、下面写一个最简单的例子,定义一个message BaseData,并对其进行扩展:
Example.proto:
// 定义一个message BaseData,100~199之间的tag可供扩展
message BaseData {
required int32 code = 1;
extensions 100 to 199;
}
// 扩展BaseData,加上一个extend_data,tag为100
extend BaseData {
required string extend_data = 100;
}
运行protoc.exe --java_out=. Example.proto生成Example.java文件,将这个Java文件加入到项目中。
Java代码:
public static void main(String[] args) throws IOException {
Example.BaseData.Builder baseBuilder = Example.BaseData.newBuilder();
baseBuilder.setCode(123);
baseBuilder.setExtension(Example.extendData, "xxg");
Example.BaseData baseData = baseBuilder.build();
// -------------- 分割线:上面是发送方,将数据序列化后发送 ---------------
ByteArrayOutputStream output = new ByteArrayOutputStream();
baseData.writeTo(output);
byte[] byteArray = output.toByteArray();
ByteArrayInputStream input = new ByteArrayInputStream(byteArray);
// -------------- 分割线:下面是接收方,将数据接收后反序列化 ---------------
ExtensionRegistry registry = ExtensionRegistry.newInstance();
registry.add(Example.extendData); // 或者Example.registerAllExtensions(registry);
Example.BaseData receiveBaseData = Example.BaseData.parseFrom(input, registry);
System.out.println(receiveBaseData.getCode());
System.out.println(receiveBaseData.getExtension(Example.extendData));
}
输出:
123
xxg
这里序列化的时候需要调用setExtension方法来设置扩展的extendData。setExtension有两个参数,第一个定义扩展的是哪一个字段,Example.extendData即表示Example类中的extendData,第二个就是设置扩展字段的值。
反序列化时,一定要对extensions进行注册。定义一个ExtensionRegistry对象registry,用registry.add(Example.extendData)或者Example.registerAllExtensions(registry)注册extensions,再将registry传入parseFrom方法作为第二个参数。如果直接按下面的方式解析就拿不到extensions的数据:
// -------------- 分割线:下面是接收方,将数据接收后反序列化 ---------------
Example.BaseData receiveBaseData = Example.BaseData.parseFrom(input);
System.out.println(receiveBaseData.getCode());
System.out.println(receiveBaseData.getExtension(Example.extendData));
输出:
123
2、另外还有一种常用的扩展定义方式:
Example.proto:
// 定义一个message BaseData,100~199之间的tag可供扩展
message BaseData {
required int32 code = 1;
extensions 100 to 199;
}
// 定义一个message Data
message Data {
required string msg = 1;
// 扩展BaseData,加上一个Data类型的字段,tag为100
extend BaseData {
required Data extend_data = 100;
}
}
这种方式看起来有点奇怪,实际上可以理解成这样:
// 定义一个message BaseData,100~199之间的tag可供扩展
message BaseData {
required int32 code = 1;
extensions 100 to 199;
}
// 定义一个message Data
message Data {
required string msg = 1;
}
// 扩展BaseData,加上一个Data类型的字段,tag为100
extend BaseData {
required Data extend_data = 100;
}
Java代码:
public static void main(String[] args) throws IOException {
// 先构造一个message Data类型对象
Example.Data.Builder dataBuilder = Example.Data.newBuilder();
dataBuilder.setMsg("xxg");
Example.Data data = dataBuilder.build();
// 再构造message BaseData对象,将Data对象通过setExtension设置到BaseData中
Example.BaseData.Builder baseBuilder = Example.BaseData.newBuilder();
baseBuilder.setCode(123);
baseBuilder.setExtension(Example.Data.extendData, data);
Example.BaseData baseData = baseBuilder.build();
// -------------- 分割线:上面是发送方,将数据序列化后发送 ---------------
ByteArrayOutputStream output = new ByteArrayOutputStream();
baseData.writeTo(output);
byte[] byteArray = output.toByteArray();
ByteArrayInputStream input = new ByteArrayInputStream(byteArray);
// -------------- 分割线:下面是接收方,将数据接收后反序列化 ---------------
ExtensionRegistry registry = ExtensionRegistry.newInstance();
registry.add(Example.Data.extendData); // 或者Example.registerAllExtensions(registry);
Example.BaseData receiveBaseData = Example.BaseData.parseFrom(input, registry);
System.out.println(receiveBaseData.getCode());
System.out.println(receiveBaseData.getExtension(Example.Data.extendData).getMsg());
}
输出:
123
xxg
作者:叉叉哥 转载请注明出处:http://blog.csdn.net/xiao__gui/article/details/37874081