java protobuf 反射_protobuf在java应用中通过反射动态创建对象(DynamicMessage)

本文介绍了如何在Java中使用protobuf通过反射动态创建对象,解决在接收客户端不同数据类型请求时的困扰。文章讨论了两种方法,包括使用单一protobuf文件和动态消息类DynamicMessage。重点讲述了利用DynamicMessage结合反射技术,根据操作码动态解析protobuf消息,实现灵活的服务器端处理。
摘要由CSDN通过智能技术生成

---恢复内容开始---

最近编写一个游戏用到protobuf数据格式进行前后台传输,苦于protobuf接受客户端的数据时是需要数据类型的如xxx.parseForm(...),这样就要求服务器在接受客户端请求时必须知道客户端传递的数据类型。由于客户端的请求数据是多种多样的,服务器端又不知道客户端的请求到底是哪个类型,这样就使得服务器端编程带来很多麻烦,甚至寸步难行。难道就没有解决办法了吗,答案当然是有的。下面就说一下常用的方法。(在看本文之前建议先了解protobuf的一些基本语法,和基本用法)

1.第一种方法也是最简单的方法,就是在整个应用程序中只定义一个proto文件,那么所有的请求都是一种类型,那么服务器端就不用苦恼怎么解析请求数据了,因为不管哪个请求数据都用同一个对象解析。如下面的列子:

首先贴一个PBMessage.proto文件

//客户端请求以及服务端响应数据协议

option java_outer_classname = "PBMessageProto";packagecom.ppsea.message;import "main/resources/message/DataMsg.proto";

message PBMessage{

optional int32 playerId= 1; //玩家id

required int32 actionCode = 2; //操作码id

optional bytes data = 5; //提交或响应的数据

optional DataMsg dataMsg = 6; //服务器端推送数据

optional string sessionKey = 7; //请求的校验码

optional int32 sessionId = 8;//当前请求的标示

}

如上述代码,整个应用都基于PBMessage.proto传输,注意到protobuf 语法中 optional修饰符,他表示这个字段是非必须的,也就是说对于客户端的不同请求,只需要为它填充其请求时用到的字段的值即可,其他的字段的值就不用管了,这样就可以模拟出各种请求来,那么接下来我们就用:PBMessage.parseForm(byte_PBMesage) //byte_PBMesage表示客户端请求数据 ,这样请求的解析就完成了。同时我们注意到: (required int32 actionCode = 2; //操作码id) ,require

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个基于Protobuf动态解析在Java的示例程序: 1. 下载protobuf-java.jar和protobuf-java-util.jar包并添加到classpath。 2. 定义一个proto文件,例如: ``` syntax = "proto3"; message Person { string name = 1; int32 age = 2; repeated string hobbies = 3; } ``` 3. 使用protoc编译proto文件,生成Java类: ``` protoc --java_out=. person.proto ``` 4. 在Java动态解析protobuf消息: ``` import com.google.protobuf.DynamicMessage; import com.google.protobuf.Descriptors; import com.google.protobuf.Descriptors.FieldDescriptor; import com.google.protobuf.ByteString; public class DynamicProtoExample { public static void main(String[] args) throws Exception { Descriptors.Descriptor descriptor = Person.getDescriptor(); // Person是生成的JavaDynamicMessage.Builder builder = DynamicMessage.newBuilder(descriptor); builder.setField(descriptor.findFieldByName("name"), "Tom"); builder.setField(descriptor.findFieldByName("age"), 18); builder.addRepeatedField(descriptor.findFieldByName("hobbies"), "reading"); builder.addRepeatedField(descriptor.findFieldByName("hobbies"), "swimming"); DynamicMessage message = builder.build(); ByteString data = message.toByteString(); DynamicMessage parsedMessage = DynamicMessage.parseFrom(descriptor, data); String name = parsedMessage.getField(descriptor.findFieldByName("name")).toString(); int age = (int) parsedMessage.getField(descriptor.findFieldByName("age")); List<String> hobbiesList = parsedMessage.getRepeatedField(descriptor.findFieldByName("hobbies")); System.out.println("Name: " + name + ", Age: " + age + ", Hobbies: " + hobbiesList); } } ``` 这个程序动态创建了一个Person消息,将它转化为字节数组,然后再将其解析回Person消息,并从获取字段值。 这是一个简单的示例,但它演示了如何使用Protobuf动态解析在Java创建和解析消息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值