欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

基于Protobuf動態(tài)解析在Java中的應(yīng)用包含例子程序

最近在做ProtoBuf相關(guān)的項目,其中用到了動態(tài)解析,網(wǎng)上看了下相關(guān)資料和博文都比較少,自己來寫一個記錄一下學(xué)習(xí)過程。

創(chuàng)新互聯(lián)是一家專注于網(wǎng)站建設(shè)、成都網(wǎng)站制作與策劃設(shè)計,荔城網(wǎng)站建設(shè)哪家好?創(chuàng)新互聯(lián)做網(wǎng)站,專注于網(wǎng)站建設(shè)十年,網(wǎng)設(shè)計領(lǐng)域的專業(yè)建站公司;建站業(yè)務(wù)涵蓋:荔城等地區(qū)。荔城做網(wǎng)站價格咨詢:18982081108

Protocol Buffers是結(jié)構(gòu)化數(shù)據(jù)格式標準,提供序列化和反序列方法,用于存儲和交換。語言中立,平臺無關(guān)、可擴展。目前官方提供了C++、Java、Python API,也有其他語言的開源api(比如php)。可通過 .proto文件生成對應(yīng)語言的類代碼
如果已知protobuf內(nèi)容對應(yīng)的是哪個類對象,則可以直接使用反序列化方法搞定(Xxx.parseFrom(inputStream)由二進制轉(zhuǎn)換,TextFormat.merge(string, xxxBuilder)由文本轉(zhuǎn)換)

而我們經(jīng)常遇到的情況是,拿到一個被protobuf序列化的二進制內(nèi)容,但不知道它的類型,無法獲得對應(yīng)的類對象。這種多見于需要處理各種各樣未知的ProtoBuf對象的系統(tǒng)。ProtoBuf提供了動態(tài)解析機制來解決這個問題,它要求提供二進制內(nèi)容的基礎(chǔ)上,再提供對應(yīng)類的Descriptor對象,在解析時通過DynamicMessage類的成員方法來獲得對象結(jié)果。
最后問題就是Descriptor對象從哪里來?這是通過protoc --descriptor_set_out=$outputpath 命令生成descriptor文件,進而得到的。

代碼如下:

 cinema.proto

option java_package="com.liulei.cinema";

enum MovieType{
 CHILDREN=1;
 ADULT=2;
 NORMAL=3;
 OHTER=4;
}

enum Gender{
 MAN=1;
 WOMAN=2;
 OTHER=3;
}

message Movie{
 required string name=1;
 required MovieType type=2;
 optional int32 releaseTimeStamp=3;
 optional string description=4;
}

message Customer{
 required string name=1;
 optional Gender gender=2;
 optional int32 birthdayTimeStamp=3;
}

message Ticket{
 required int32 id=1;
 required Movie movie=2;
 required Customer customer=3;
}

Main.java

public static void main( String[] args ) {

  Cinema.Movie.Builder movieBuilder = Cinema.Movie.newBuilder();
  movieBuilder.setName("The Shining");
  movieBuilder.setType(Cinema.MovieType.ADULT);
  movieBuilder.setReleaseTimeStamp(327859200);

  System.out.println("Dynamic Message Parse by proto file");
  try {
   byte[] buffer3 = new byte[movieBuilder.build().getSerializedSize()];
   CodedOutputStream codedOutputStream3 = CodedOutputStream.newInstance(buffer3);
   try {
    movieBuilder.build().writeTo(codedOutputStream3);
    System.out.println(buffer3);
   } catch (IOException e) {
    e.printStackTrace();
   }
   String protocCMD = "protoc --descriptor_set_out=cinema.description ./cinema.proto --proto_path=.";
   Process process = Runtime.getRuntime().exec(protocCMD);
   process.waitFor();
   int exitValue = process.exitValue();
   if (exitValue != 0) {
    System.out.println("protoc execute failed");
    return;
   }
   Descriptors.Descriptor pbDescritpor = null;
   DescriptorProtos.FileDescriptorSet descriptorSet = DescriptorProtos.FileDescriptorSet.parseFrom(new FileInputStream("./cinema.description"));
   for (DescriptorProtos.FileDescriptorProto fdp : descriptorSet.getFileList()) {
    Descriptors.FileDescriptor fileDescriptor = Descriptors.FileDescriptor.buildFrom(fdp, new Descriptors.FileDescriptor[]{});
    for (Descriptors.Descriptor descriptor : fileDescriptor.getMessageTypes()) {
     if (descriptor.getName().equals("Movie")) {
      System.out.println("Movie descriptor found");
      pbDescritpor = descriptor;
      break;
     }
    }
   }
   if (pbDescritpor == null) {
    System.out.println("No matched descriptor");
    return;
   }
   DynamicMessage.Builder pbBuilder = DynamicMessage.newBuilder(pbDescritpor);

   Message pbMessage = pbBuilder.mergeFrom(buffer3).build();
   System.out.println(pbMessage);

  } catch (Exception e) {
   System.out.println("Exception");
   e.printStackTrace();
  }
 }

執(zhí)行結(jié)果:

Dynamic Message Parse From byte array
[B@597ccf6e
Movie descriptor found
name: "The Shining"
type: ADULT
releaseTimeStamp: 327859200

 解釋具體過程:

0.首先對.proto文件使用protoc命令,生成的descriptor文件中包含多個類對應(yīng)的descriptor類信息(序列化的DescriptorSet內(nèi)容)

1.首先取出序列化的DescriptorSet內(nèi)容,F(xiàn)ileDescriptorSet.parseFrom方法反序列化得到FileDescriptorSet對象

2.取出對應(yīng)message類型的Descriptor。

 DescriptorSet成員方法getFileList(),拿到多個FileDescriptorProto對象,再構(gòu)建對應(yīng)FileDescriptor。
 FileDescriptor的成員方法getMessageTypes()得到所有Message的Descriptor對象,找到對應(yīng)名字的Descriptor

3.用Descriptor對象反序列化對象

構(gòu)建DynamicMessage.Builder對象builder,再調(diào)用builder的mergeFrom/merge方法得到Message對象

其中Descriptor相關(guān)類:

DescriptorProtos.DescriptorSet:protoc編譯出來類文件中包含這個類,描述多個.proto文件中的類

DescriptorProtos.FileDescriptorProto:描述一個完整的.proto文件中的類

DescriptorProtos.FileDescriptor:由DescriptorProtos.FileDescriptorProto構(gòu)建而來(buildFrom),描述1個完整.proto文件中的所有內(nèi)容,包括message類型的Descriptor和其他被導(dǎo)入文件的Descriptor。

getMessageTypes()方法:返回List<Descriptors.Descriptor>。得到FileDescriptor內(nèi),所有message類型直接兒子的Descriptor列表   

DescriptorProtos.Descriptor:描述一個message類型,通過getName()得到message的類名

以上這篇基于Protobuf動態(tài)解析在Java中的應(yīng)用 包含例子程序就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持創(chuàng)新互聯(lián)。

分享名稱:基于Protobuf動態(tài)解析在Java中的應(yīng)用包含例子程序
文章地址:http://aaarwkj.com/article26/jeghcg.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供商城網(wǎng)站網(wǎng)站內(nèi)鏈、建站公司、定制網(wǎng)站、做網(wǎng)站、用戶體驗

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

外貿(mào)網(wǎng)站制作
夫妻性生活免费看视频| 国产在线观看国产精品| 国产传媒视频网站在线观看| 一本色道久久亚洲综合精品蜜桃| 蜜桃一区二区三区免费| 亚洲综合色一区二区三区四区| 亚洲精品国产自在现线| 成人免费在线观看午夜| 日本熟女肥臀一区二区| 欧美亚洲少妇人妻系列| 日本一区二区精美视频| 欧美日韩国产另类在线视频| 久久亚洲精品中文字幕| 亚洲av成人在线不卡| 国产av剧情在线免费观看| 国产一区av麻豆免费观看| 日韩av高清在线免费观看| 亚洲av乱码一区二区三四五六七| 在线免费观看视频97| 五月婷婷丁香噜噜噜噜| 国产精品三级一区二区三区| 91日韩人妻一区二区三区| 国产av无毛一区二区三区| 韩国黄色理论片一区二区麻豆| 欧美日韩一区二区三区色| 国产91美女黄色在线观看| 蜜臀在线免费观看av| 日韩成人三级一区二区| 日韩亚洲在线中文字幕| 麻豆国产精品原创av男女| 日本国产福利视频在线观看| 亚洲第一区二区国产精品| 国产高清av免费在线播放| 日本一区二区三区免费看视频| 免费人成网站视频在线观看不卡 | 老熟妇奂伦一区二区三区| 日本久久91跳蛋视频| 91精品国产人妻女教师| 国产精品一区二区综合亚洲| av全欧国男人在线天堂| 毛片成人18毛片免费看|