protobuf extensions(extend)的消息定义及Java使用方法

本文介绍了如何在protobuf中使用extensions对原始消息类型进行扩展,以实现消息定义的复用。通过示例展示了如何在Example.proto中定义message BaseData并扩展,以及在Java中如何使用setExtension方法进行序列化和反序列化。强调了反序列化时需注册extensions,并提供了两种不同的扩展定义方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在定义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

注意上面的Java代码用的是第一种proto消息定义方式,如果用的是下面第二种消息定义的方式,setExtension的参数Example.Data.extendData就应该改成Example.extendData,注意嵌套结构。


作者:叉叉哥   转载请注明出处:http://blog.csdn.net/xiao__gui/article/details/37874081



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值