【Protobuf】Protobuf快速使用 Java版、Python版

news2025/1/12 3:52:10

【Protobuf】Protobuf快速使用 Java版、Python版

  • Protobuf介绍
  • 快速使用(Java版)
    • 创建 .proto文件,定义数据结构
    • 安装Protobuf编译器(二选一)
    • 使用IDEA编译(二选一)
    • 使用编译后的文件
  • 快速使用(Python版)
    • 创建 .proto文件,定义数据结构
    • 安装Protobuf编译器
    • 使用编译后的Python文件
  • Protobuf中的Message语法规范

Protobuf介绍

  Protocol Buffers(简称为ProtoBuf)是一种用于序列化、结构化数据的语言无关、平台无关、可扩展的机制。它由Google开发并于2008年开源发布。

  Protocol Buffers通过使用结构化的消息定义文件(.proto文件)来描述数据结构,然后利用特定的编译器将这些消息定义文件编译成相应编程语言的类。这些生成的类提供了一组API,用于在不同的平台和编程语言之间进行数据的序列化、传输和反序列化操作。

  Protobuf支持生成代码的语言包括JavaPythonC++GoRubyC#

  官网地址:https://developers.google.com/protocol-buffers/。
  官方开源地址 :https://github.com/protocolbuffers/protobuf。
  语法指南:https://developers.google.com/protocol-buffers/docs/proto。

  Portobuf的序列化的结果体积要比XMLJSON小很多,XMLJSON的描述信息比较多,而且是文本存储格式,导致消息要大;Portobuf的存储格式是二进制。此外Portobuf还使用了Varint 编码,减少数据对空间的占用。

  Portobuf序列化和反序列化速度比XMLJSON快很多,是直接把对象和字节数组做转换,而XMLJSON还需要构建成XML或者JSON对象结构。

  但是Portobuf自解耦性差,以二进制数据流方式存储(不可读),需要通过.proto文件才能了解到数据结构。

快速使用(Java版)

创建 .proto文件,定义数据结构

  使用ProtoBuf,首先需要通过ProtoBuf语法定义数据结构(消息),这些定义好的数据结构保存在.proto为后缀的文件中。

   server.proto

// 指定protobuf的版本,proto3是最新的语法版本
syntax = "proto3";

// 定义数据结构message
message Server{
  string host = 1;   // 定义一个string类型的字段,字段名字为host, 序号为1
  int32 code = 2;   // 定义一个int32类型的字段,字段名字为code , 序号为2
}

proto文件中,字段后面的序号,不能重复,定义了就不能修改,可以理解成字段的唯一ID。

安装Protobuf编译器(二选一)

  Protobufgithub发布地址:https://github.com/protocolbuffers/protobuf/releases

  Protobuf的编译器叫protoc,在上面的网址中找到最新版本的安装包,下载安装。

在这里插入图片描述

  这里下载的是:protoc-23.2-win64.zip,下载后,解压到你想要的安装目录即可。

可以把protoc下的bin目录路径添加到环境变量,否则要输入全部bin目录写命令操作。

  protoc编译器支持将proto文件编译成多种语言版本的代码,这里以java为例。

  切换到proto文件所在的目录, 执行下面命令

protoc --java_out=. server.proto

  然后在当前目录生成了一个Server.javajava类文件,这个就是我们刚才用protobuf语法定义的数据结构对应的java类文件,通过这个类文件我们就可以操作定义的数据结构。

使用IDEA编译(二选一)

  引入pom依赖

<dependency>
    <groupId>com.google.protobuf</groupId>
    <artifactId>protobuf-java</artifactId>
    <version>3.6.1</version>
</dependency>

  设置build代码

<build>
     <extensions>
         <extension>
             <groupId>kr.motd.maven</groupId>
             <artifactId>os-maven-plugin</artifactId>
             <version>1.5.0.Final</version>
         </extension>
     </extensions>
     <plugins>
         <plugin>
             <groupId>org.xolstice.maven.plugins</groupId>
             <artifactId>protobuf-maven-plugin</artifactId>
             <version>0.5.0</version>
             <configuration>
                 <!--suppress UnresolvedMavenProperty -->
                 <protocArtifact>com.google.protobuf:protoc:3.6.1:exe:${os.detected.classifier}</protocArtifact>
                 <pluginId>grpc-java</pluginId>
                 <!--suppress UnresolvedMavenProperty -->
                 <pluginArtifact>io.grpc:protoc-gen-grpc-java:1.14.0:exe:${os.detected.classifier}</pluginArtifact>
             </configuration>
             <executions>
                 <execution>
                     <goals>
                         <goal>compile</goal>
                         <goal>compile-custom</goal>
                     </goals>
                 </execution>
             </executions>
         </plugin>
     </plugins>
 </build>

  配置好以后,将对应的proto文件放到新创建的proto文件下,在maven中执行compile/protobuf就可以,在target中我们可以找到对应的pb文件。

在这里插入图片描述

在这里插入图片描述

使用编译后的文件

  记得导入protobuf基础类库:

<dependency>
    <groupId>com.google.protobuf</groupId>
     <artifactId>protobuf-java</artifactId>
     <version>3.9.1</version>
</dependency>

  Protocol Buffers把生成的Server类复制到当前目录,然后写个demo

Server.Response.Builder builder = Server.Response.newBuilder();
// 设置字段值
builder.setHost("localhost");
builder.setCode(200);

 Server.Response response = builder.build();
 // 将数据根据protobuf格式,转化为字节数组
 byte[] byteArray  = response.toByteArray();

// 反序列化,二进制数据
try {
    Server.Response newResponse = Server.Response.parseFrom(byteArray);
    System.out.println(newResponse.getHost());
    System.out.println(newResponse.getCode());
} catch (Exception e) {

}

快速使用(Python版)

创建 .proto文件,定义数据结构

  使用ProtoBuf,首先需要通过ProtoBuf语法定义数据结构(消息),这些定义好的数据结构保存在.proto为后缀的文件中。

   server.proto

// 指定protobuf的版本,proto3是最新的语法版本
syntax = "proto3";

// 定义数据结构message
message Server{
  string host = 1;   // 定义一个string类型的字段,字段名字为host, 序号为1
  int32 code = 2;   // 定义一个int32类型的字段,字段名字为code , 序号为2
}

proto文件中,字段后面的序号,不能重复,定义了就不能修改,可以理解成字段的唯一ID。

安装Protobuf编译器

  Protobufgithub发布地址:https://github.com/protocolbuffers/protobuf/releases

  Protobuf的编译器叫protoc,在上面的网址中找到最新版本的安装包,下载安装。

在这里插入图片描述

  这里下载的是:protoc-23.2-win64.zip,下载后,解压到你想要的安装目录即可。

可以把protoc下的bin目录路径添加到环境变量,否则要输入全部bin目录写命令操作。

  protoc编译器支持将proto文件编译成多种语言版本的代码,这里以java为例。

  切换到proto文件所在的目录, 执行下面命令

protoc --python_out=. server.proto

  然后在当前目录生成了一个server_pb2.pypython文件,这个就是我们刚才用protobuf语法定义的数据结构对应的python文件,通过这个文件我们就可以操作定义的数据结构。

使用编译后的Python文件

  记得把server.protoserver_pb2.py放入一个目录下,然后写python代码访问Server

import server_pb2   # 导入生成的Python文件
from google.protobuf import json_format   # 导入protobuf的JSON格式转换库(可选)

# 创建一个新的消息对象
message = server_pb2.Response()

# 设置字段值
message.host = "172.0.0.1"
message.code = 100

# 将消息序列化为字节串
serialized_data = message.SerializeToString()

# 将字节串反序列化为消息
deserialized_message = server_pb2.Response()
deserialized_message.ParseFromString(serialized_data)

# 将消息对象转换为JSON字符串
json_string = json_format.MessageToJson(message)

# 将JSON字符串转换为消息对象
json_message = server_pb2.Response()

print(json_format.Parse(json_string, json_message))

Protobuf中的Message语法规范

  参考下一篇:点击查看

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/629145.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Spring源码解读三】IoC容器之AnnotationConfigApplication的refresh()刷新方法其二

invokeBeanFactoryPostProcessors() PriorityOrdered接口 Ordered接口 invokeBeanDefinitionRegistryPostProcessors() registerBeanPostProcessors() getBeanNamesForType() initMessageSource() initApplicationEventMulticaster() onRefresh() registerListeners()…

听我一句劝,别去外包,干了三年,真废了....

先说一下自己的情况&#xff0c;大专生&#xff0c;18年通过校招进入湖南某软件公司&#xff0c;干了接近4年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落!而我已经在一个企业干了四年的功能测试…

找不到xinput1_3.dll怎么办?xinput1_3.dll丢失的四个修复方法

在我们打开游戏的或者软件的时候&#xff0c;电脑提示“找不到xinput1_3.dll&#xff0c;无法继续执行此代码”怎么办&#xff1f;相信困扰着不少小伙伴&#xff0c;我再在打开吃鸡的时候&#xff0c;然后花了一上午的时候时间研究&#xff0c;现在终于知道xinput1_3.dll文件是…

Windows10系统开启 Telnet客户端 功能

1、应用场景 在实际工作中&#xff0c;经常有查看机器端口连通性的场景&#xff08;主要为了确认某台机器上的服务是否正常&#xff0c;比如&#xff1a;查看的端口 9091&#xff09; telnet 192.168.166.159 9091如下状态说明telnet 端口是通的 Ctrl ] ,退出Telnet连接&a…

探究Vue源码:mustache模板引擎(2) mustache使用方法

mustache是最早的模板引擎 比vue的诞生还要早很多 而他的语法 就是基于 {{ }} 这样的双花括号 mustache属于通用性的应用 他既可以在浏览器中直接用 也可以在npm中使用 这里 我们为了方便 就直接去拿在浏览器中使用的包了 没必要再自己搭个环境 大家可以下载我上传的资源 vue源…

职场老油条表示真干不过,部门新来的00后测试员已把我卷崩溃,想离职了...

在程序员职场上&#xff0c;什么样的人最让人反感呢? 是技术不好的人吗?并不是。技术不好的同事&#xff0c;我们可以帮他。 是技术太强的人吗?也不是。技术很强的同事&#xff0c;可遇不可求&#xff0c;向他学习还来不及呢。 真正让人反感的&#xff0c;是技术平平&#x…

Linux学习之vim正常模式和插入模式

使用vim新建或者打开一个文件后&#xff0c;首先进入的就是正常模式&#xff0c;从正常模式按不同的按键能够进入其他三种模式。 在正常模式下&#xff0c;按i&#xff0c;I&#xff08;大写的i键&#xff09;&#xff0c;a&#xff0c;A&#xff0c;o&#xff08;小写的o&…

【3DsMAX】从零开始建房(6)

目录 1. 制作广告牌 2. 制作屋顶小船船身 1. 制作广告牌 先创建一个长方体 转换为可编辑多边形&#xff0c;选中面&#xff0c;插入 挤出 添加两个圆柱体作为支架 用轮廓工具收一下面 选中这三个物体打组 统一材质 设置线条颜色为黑色 2. 制作屋顶小船船身 先添加一个球体&…

顺序表刷题(1~3)

目录 移除元素 删除有序数组重复项 合并有序数组 移除元素 方法一&#xff1a; 如果找到一个删除一个这样的时间复杂度为O(n^2)&#xff08;最坏删除所有数据&#xff09;删除后还要挪动数据。我们可以将符合条件的数组元素放入一个临时数组中&#xff0c;这种方法的时间复杂…

什么是数据结构

一、什么是数据结构 1、数据结构的定义 数据&#xff1a;从计算机的角度来看&#xff0c;数据是所有能被输入到计算机中且能被计算机处理的符号的集合。它是计算机操作的对象的总称&#xff0c;也是计算机处理信息的某种特定的符号表示形式&#xff08;二进制码的抽象表示&am…

【高级篇】服务异步通信

服务异步通信-高级篇 消息队列在使用过程中&#xff0c;面临着很多实际问题需要思考&#xff1a; 1.消息可靠性 消息从发送&#xff0c;到消费者接收&#xff0c;会经理多个过程&#xff1a; 其中的每一步都可能导致消息丢失&#xff0c;常见的丢失原因包括&#xff1a; 发送…

030 JavaWeb Html CSS

目录 JavaWeb概述1.访问web的原理2.C/S软件和B/S软件区别3.静态网站和动态网站 HTMLHTML的概述Table表格详细用法见W3CSchool.chm合并单元格课程表 img标签table和img标签组合使用a标签表单表单Get提交和post提交 div和span CSS1.CSS概述2.CSS语法3.CSS三种写法行内样式内部样式…

VueX使用简明笔记

1、作用&#xff1a; vuex是使用vue中必不可少的一部分&#xff0c;基于父子、兄弟组件&#xff0c;我们传值可能会很方便&#xff0c;但是如果是没有关联的组件之间要使用同一组数据&#xff0c;就显得很无能为力&#xff0c;那么vuex就很好的解决了我们这种问题&#xff0c;…

Knife4j的使用、SpringFox和SpringDoc介绍

knife4j是一个Swagger的增强工具&#xff0c;能够完善项目的接口文档。 官网&#xff1a; Knife4j 集Swagger2及OpenAPI3为一体的增强解决方案. | Knife4j Swagger的生成的默认文档确实不好用&#xff08;不美观、不支持搜索、不能导出&#xff09; 在Swagger2中使用 想一想…

Android 12.0 原生SystemUI下拉通知栏每条通知默认展开

1.前言 在12.0的系统rom原生开发中,在在对SystemUI下拉通知栏做定制的时候,在下拉状态栏的时候,通知栏中 最后一条通知默认是收缩的 点击按钮 就会展开 原生系统systemui就是如此,为了更美观 所以要求最后一条通知也默认展开,显得更美观 最终效果图: 2.原生SystemUI下拉通…

C++/C按照时间命名保存bin文件

背景 在Linux应用编程过程中&#xff0c;使用C或者C语言保存、读取bin文件是比较常见的需求。这里详细记录一下使用C保存bin文件&#xff0c;也可以使用C语言实现。 代码 C/C语言保存bin文件函数&#xff0c;C中也能使用 正确写入返回0&#xff0c;错误返回-1 // C 保存bi…

双重检查锁定与延迟初始化

双重检查锁定的由来 在Java程序中&#xff0c;有时候可能需要推迟一些高开销的对象初始化操作&#xff0c;并且只有在使用这些对象时才进行初始化。此时&#xff0c;程序员可能会采用延迟初始化。但要正确实现线程安全的延迟初始化需要一些技巧&#xff0c;否则很容易出现问题…

51单片机操作系统——RTX51 Tiny

简介 RTX51 是keil公司开发的一款实时操作系统&#xff0c;其有两个版本&#xff1a; 1.Tiny 2.Full&#xff0c;区别如下&#xff1a; RTX51 Full &#xff1a;使用四个任务优先权完成同时存在时间片轮转调度和抢先的任务切换 RTX51工作与中断功能相似的状态下 &#xff0c…

华为OD机试真题 JavaScript 实现【最远足迹】【2022Q4 100分】,附详细解题思路

一、题目描述 某探险队负责对地下洞穴进行探险。探险队成员在进行探险任务时&#xff0c;随身携带的记录器会不定期地记录自身的坐标&#xff0c;但在记录的间隙中也会记录其他数据。探索工作结束后&#xff0c;探险队需要获取到某成员在探险过程中相对于探险队总部的最远的足…

【打卡】苹果叶片病害分类和建筑物变化检测数据挖掘竞赛

【打卡】苹果叶片病害分类和建筑物变化检测数据挖掘竞赛 文章目录 【打卡】苹果叶片病害分类和建筑物变化检测数据挖掘竞赛Task 1两个赛题数据可视化任务2 苹果病害数据加载与数据增强任务三 Task 1两个赛题数据可视化 在这个任务中&#xff0c;参赛选手需要对两个赛题的数据进…