记录:Protocol Buffers(protobuf)在Java开发中使用

简介:


    1.编写一个.proto文件命名为:addressbook.proto,该文件内容来自protocal-buffers官网

    

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
package tutorial;
 
option java_package = "com.example.tutorial";
option java_outer_classname = "AddressBookProtos";
 
message Person {
   required string name = 1;
   required int32 id = 2;
   optional string email = 3;
 
   enum PhoneType {
     MOBILE = 0;
     HOME = 1;
     WORK = 2;
   }
 
   message PhoneNumber {
     required string number = 1;
     optional PhoneType type = 2 [default = HOME];
   }
 
   repeated PhoneNumber phone = 4;
}
 
message AddressBook {
   repeated Person person = 1;
}

    

   2.使用protoc-2.6.0-win32.zip解压后的protoc.exe生成Java类

    查询protoc.exe帮助:

   

1
D:\__dev\jar\protobuff>protoc.exe -h

    生成Java类:

   

1
2
D:\__dev\jar\protobuff>protoc.exe --proto_path=F:\__eclipse\test\proto 
--java_out=F:\__eclipse\test\src F:\__eclipse\test\proto\addressbook.proto

   

 在Eclipse中的项目目录结构图如下:

 wKioL1Q7ut2TgBcxAAEKLAg_voM830.jpg


说明:

上图中:addressbook.proto数据格式文件,AddressBookProtos.java是生成的java类,protobuf-java-2.5.0.jar是Java运行时类库。


3.使用AddressBookProtos类来实现对象的序列号和反序列化,了解Protocol-buffers jar的相关API

  代码实例:

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
package  com.example.test;
 
import  java.util.Arrays;
 
import  com.example.tutorial.AddressBookProtos.AddressBook;
import  com.example.tutorial.AddressBookProtos.Person;
import  com.google.protobuf.InvalidProtocolBufferException;
 
public  class  AddressBookProtoUse {
 
     public  static  void  main(String[] args) {
         //构建一个Person对象
         Person person = Person
                 .newBuilder()
                 .setEmail( "zhangsan@163.com" )
                 .setId( 10086 )
                 .setName( "zhangsan" )
                 .addPhone(
                         Person.PhoneNumber.newBuilder().setNumber( "186" )
                                 .setType(Person.PhoneType.HOME).build())
                 .build();
         System.out.println( "打印输出Person对象信息:" );
         System.out.println(person);
         System.out.println( "Person对象调用toString()方法:" );
         System.out.println(person.toString());
 
         System.out.println( "Person对象字段是否初始化:"  + person.isInitialized());
 
         // 序列号
         System.out.println( "Person对象调用toByteString()方法:" );
         System.out.println(person.toByteString());
 
         System.out.println( "Person对象调用toByteArray()方法:" );
         System.out.println(Arrays.toString(person.toByteArray()));
         
         try  {
             System.out.println( "反序列化后的对象信息:" );
             // 反序列化
             Person newPerson = Person.parseFrom(person.toByteArray());
             System.out.println(newPerson);
             newPerson = Person.parseFrom(person.toByteString());
             System.out.println(newPerson);
         catch  (InvalidProtocolBufferException e) {
             e.printStackTrace();
         }
 
         // 向地址簿添加两条Person信息
         AddressBook.Builder books = AddressBook.newBuilder();
         books.addPerson(person);
         books.addPerson(Person.newBuilder(person).setEmail( "tom@163.com" )
                 .build());
         System.out.println( "AddressBook对象信息:" );
         System.out.println(books.build());
 
     }
}


 运行结果:

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
打印输出Person对象信息:
name: "zhangsan"
id: 10086
email: "zhangsan@163.com"
phone {
   number: "186"
   type: HOME
}
 
Person对象调用toString()方法:
name: "zhangsan"
id: 10086
email: "zhangsan@163.com"
phone {
   number: "186"
   type: HOME
}
 
Person对象字段是否初始化:true
Person对象调用toByteString()方法:
<ByteString@89ec59 size=40>
Person对象调用toByteArray()方法:
[10, 8, 122, 104, 97, 110, 103, 115, 97, 110, 16, -26, 78, 26, 16, 122, 104, 97, 110, 103, 115, 97, 110, 64, 49, 54, 51, 46, 99, 111, 109, 34, 7, 10, 3, 49, 56, 54, 16, 1]
反序列化后的对象信息:
name: "zhangsan"
id: 10086
email: "zhangsan@163.com"
phone {
   number: "186"
   type: HOME
}
 
name: "zhangsan"
id: 10086
email: "zhangsan@163.com"
phone {
   number: "186"
   type: HOME
}
 
AddressBook对象信息:
person {
   name: "zhangsan"
   id: 10086
   email: "zhangsan@163.com"
   phone {
     number: "186"
     type: HOME
   }
}
person {
   name: "zhangsan"
   id: 10086
   email: "tom@163.com"
   phone {
     number: "186"
     type: HOME
   }
}


 探究AddressBookProtos类:

 wKioL1Q7vpbCMh3gAAIqxencT_Y283.jpg


 a.构造Person对象,Person类继承自com.google.protobuf.GeneratedMessage类,而GeneratedMessage类继承自AbstractMessage类,并且实现了序列化接口Serializable。在AbstractMessage类中重写了toString()方法,具体内容如下:

  

1
2
3
4
@Override
   public  final  String toString() {
     return  TextFormat.printToString( this );
   }

 于是有了Person对象调用toString()方法后直接输出Person对象的文本内容。

 b.toByteString()返回的ByteString是一个不可变的byte序列,由AbstractMessage类实现。toByteArray()返回byte[]。这两个方法都是对象进行序列化的方法。


 c.isInitialized()判断对象的字段是否初始化,该方法与Person类的initFields()方法相关。

   initFields()源代码:

  

1
2
3
4
5
6
private  void  initFields() {
       name_ =  "" ;
       id_ =  0 ;
       email_ =  "" ;
       phone_ = java.util.Collections.emptyList();
     }

 

  initFields()方法的调用是在Person类的默认实力对象初始化之后调用的,在Person类的静态代码块中可以看到:

 

1
2
3
4
static  {
       defaultInstance =  new  Person( true );
       defaultInstance.initFields();
     }


 d.Person类提供了一系列的反序列化的重载方法用来讲数据反序列化为Person对象。

   wKioL1Q7wjHQC9n2AAFzEqM48pQ285.jpg


 e.关于Person对象的Build

   Person类中有一个Builder的内部类,该类用来构建Person对象,并且为Person对象添加数据。

  

1
2
3
public  static  final  class  Builder  extends
         com.google.protobuf.GeneratedMessage.Builder<Builder>
        implements  com.example.tutorial.AddressBookProtos.PersonOrBuilder

  

4.通过学习官网的实例,手绘一张Java使用protobuf的基本流程图

  wKiom1Q7xw-yhSTsAAGqSK_eyQE678.jpg



本文转自 secondriver 51CTO博客,原文链接:http://blog.51cto.com/aiilive/1563449,如需转载请自行联系原作者

相关文章
|
1月前
|
安全 前端开发 Java
《深入理解Spring》:现代Java开发的核心框架
Spring自2003年诞生以来,已成为Java企业级开发的基石,凭借IoC、AOP、声明式编程等核心特性,极大简化了开发复杂度。本系列将深入解析Spring框架核心原理及Spring Boot、Cloud、Security等生态组件,助力开发者构建高效、可扩展的应用体系。(238字)
|
4月前
|
人工智能 前端开发 Java
2025年WebStorm高效Java开发全指南:从配置到实战
WebStorm 2025不仅是一款强大的JavaScript IDE,也全面支持Java开发。本文详解其AI辅助编程、Java特性增强及性能优化,并提供环境配置、高效开发技巧与实战案例,助你打造流畅的全栈开发体验。
478 4
|
4月前
|
前端开发 JavaScript Java
Java 开发中 Swing 界面嵌入浏览器实现方法详解
摘要:Java中嵌入浏览器可通过多种技术实现:1) JCEF框架利用Chromium内核,适合复杂网页;2) JEditorPane组件支持简单HTML显示,但功能有限;3) DJNativeSwing-SWT可内嵌浏览器,需特定内核支持;4) JavaFX WebView结合Swing可完美支持现代网页技术。每种方案各有特点,开发者需根据项目需求选择合适方法,如JCEF适合高性能要求,JEditorPane适合简单展示。(149字)
542 1
|
4月前
|
安全 Java 领域建模
Java 17 探秘:不容错过的现代开发利器
Java 17 探秘:不容错过的现代开发利器
415 0
|
2月前
|
消息中间件 人工智能 Java
抖音微信爆款小游戏大全:免费休闲/竞技/益智/PHP+Java全筏开源开发
本文基于2025年最新行业数据,深入解析抖音/微信爆款小游戏的开发逻辑,重点讲解PHP+Java双引擎架构实战,涵盖技术选型、架构设计、性能优化与开源生态,提供完整开源工具链,助力开发者从理论到落地打造高留存、高并发的小游戏产品。
|
2月前
|
存储 Java 关系型数据库
Java 项目实战基于面向对象思想的汽车租赁系统开发实例 汽车租赁系统 Java 面向对象项目实战
本文介绍基于Java面向对象编程的汽车租赁系统技术方案与应用实例,涵盖系统功能需求分析、类设计、数据库设计及具体代码实现,帮助开发者掌握Java在实际项目中的应用。
125 0
|
3月前
|
JavaScript 安全 前端开发
Java开发:最新技术驱动的病人挂号系统实操指南与全流程操作技巧汇总
本文介绍基于Spring Boot 3.x、Vue 3等最新技术构建现代化病人挂号系统,涵盖技术选型、核心功能实现与部署方案,助力开发者快速搭建高效、安全的医疗挂号平台。
233 3
|
3月前
|
安全 Java 数据库
Java 项目实战病人挂号系统网站设计开发步骤及核心功能实现指南
本文介绍了基于Java的病人挂号系统网站的技术方案与应用实例,涵盖SSM与Spring Boot框架选型、数据库设计、功能模块划分及安全机制实现。系统支持患者在线注册、登录、挂号与预约,管理员可进行医院信息与排班管理。通过实际案例展示系统开发流程与核心代码实现,为Java Web医疗项目开发提供参考。
210 2