首页 > 其他分享 >Protobuf vs JSON

Protobuf vs JSON

时间:2023-08-13 14:55:35浏览次数:44  
标签:rand Protobuf proto JSON vs NewString uuid

Protobuf(Protocol Buffers)和 JSON 都是数据序列化格式,但它们在许多方面有着显著的不同。以下是对两者的一些主要比较:

  1. 数据大小和速度:
    • Protobuf:由于 Protobuf 是二进制格式,因此它生成的数据通常比 JSON 小很多,这使得 Protobuf 在网络传输中更加高效。同时,Protobuf 的解析和序列化速度也比 JSON 快。
    • JSON:JSON 是文本格式,它生成的数据通常比 Protobuf 大,且解析和序列化速度较慢。
  2. 可读性和易用性:
    • Protobuf:Protobuf 是二进制格式,人类无法直接阅读。此外,使用 Protobuf 需要预先定义数据结构(.proto 文件),这增加了使用的复杂性。
    • JSON:JSON 是文本格式,人类可以直接阅读和编辑。此外,JSON 的数据结构可以在运行时动态定义,这使得 JSON 更易于使用。
  3. 类型安全和版本兼容性:
    • Protobuf:Protobuf 支持静态类型检查,这可以在编译时捕获类型错误。此外,Protobuf 设计了一套版本兼容性机制,可以在不破坏旧版本的情况下添加新的字段。
    • JSON:JSON 是动态类型的,无法在编译时捕获类型错误。此外,JSON 没有内置的版本兼容性机制,如果数据结构发生变化,可能需要修改代码以适应新的结构。
  4. 支持的语言:
    • Protobuf:Google 提供了多种语言的 Protobuf 库,包括 C++、Java、Python、Golang 等。
    • JSON:几乎所有的编程语言都支持 JSON。
  5. 学习成本:
    • Protobuf:Protobuf 的学习曲线相对较陡。你需要理解 Protobuf 的语法,学习如何编写 .proto 文件,并且需要了解如何使用 Protobuf 编译器生成代码。此外,你还需要理解 Protobuf 的版本兼容性规则。
    • JSON:JSON 的学习曲线相对较平。JSON 的语法非常简单,大多数人可以在很短的时间内掌握。此外,几乎所有的编程语言都内置了 JSON 的支持,你不需要安装任何额外的库就可以开始使用 JSON。
  6. 使用成本:
    • Protobuf:Protobuf 的使用成本相对较高。首先,你需要为每个数据结构编写一个 .proto 文件,然后使用 Protobuf 编译器生成代码。此外,如果你的数据结构发生了变化,你需要更新 .proto 文件并重新生成代码。这些步骤都需要额外的时间和工作。
    • JSON:JSON 的使用成本相对较低。你可以直接在代码中定义数据结构,无需任何额外的步骤。此外,如果你的数据结构发生了变化,你只需要更新你的代码,无需任何其他操作。

总的来说,Protobuf 和 JSON 各有优劣,适用于不同的场景。如果你需要高效的数据传输和严格的类型检查,那么 Protobuf 可能是一个好选择。如果你需要易于使用和人类可读的数据格式,那么 JSON 可能更适合你。

下面以一个简单的例子来对Protobuf和JSON运行效率进行简单对比。

定义proto文件

首先通过.proto来定义所需的结构:

syntax = "proto3";

package pvsj.proto;

option go_package = "./;proto";

import "google/protobuf/struct.proto";

message Base {
  string tx_hash = 1;
  int64 timestamp = 2;
  google.protobuf.Struct extra = 3;
  uint64 block_number = 4;
  int32 category = 5;
}

message CertGen {
  string id = 1;
  string issuer = 2;
  string name = 3;
  string number = 4;
  string seal_name = 5;
  string seal_number = 6;
  string sign_hash = 7;
  string date = 8;
  Base base = 9;
}

然后通过protoc生成响应的go代码。

Benchmark

main.go内容如下:

package main

import (
	"encoding/json"
	"math/rand"
	"time"

	"github.com/bytedance/sonic"
	"github.com/google/uuid"
	pb "github.com/mengbin92/pvsj/proto"
	"google.golang.org/protobuf/proto"
	"google.golang.org/protobuf/types/known/structpb"
)

func init() {
	rand.New(rand.NewSource(time.Now().UnixNano()))
	uuid.SetRand(rand.New(rand.NewSource(time.Now().UnixNano())))
}

func main() {
}

// 使用protobuf进行序列化和反序列化
func genProto(num int) {
	for i := 0; i < num; i++ {
		data := map[string]interface{}{
			"name":  uuid.NewString(),
			"age":   rand.Int(),
			"score": rand.Float64(),
		}
		extra, _ := structpb.NewStruct(data)

		base := &pb.Base{
			TxHash:      uuid.NewString(),
			Timestamp:   time.Now().Unix(),
			Extra:       extra,
			BlockNumber: rand.Uint64(),
			Category:    rand.Int31(),
		}

		gen := &pb.CertGen{
			Id:         uuid.NewString(),
			Issuer:     uuid.NewString(),
			Name:       uuid.NewString(),
			Number:     uuid.NewString(),
			SealName:   uuid.NewString(),
			SealNumber: uuid.NewString(),
			SignHash:   uuid.NewString(),
			Date:       time.Now().Format(time.DateTime),
			Base:       base,
		}

		genBytes, _ := proto.Marshal(gen)
		proto.Unmarshal(genBytes, gen)
	}
}

// 使用sonic对json进行序列化和反序列化
func genJsonSonic(num int) {
	for i := 0; i < num; i++ {
		data := map[string]interface{}{
			"name":  uuid.NewString(),
			"age":   rand.Int(),
			"score": rand.Float64(),
		}
		extra, _ := structpb.NewStruct(data)

		base := &pb.Base{
			TxHash:      uuid.NewString(),
			Timestamp:   time.Now().Unix(),
			Extra:       extra,
			BlockNumber: rand.Uint64(),
			Category:    rand.Int31(),
		}

		gen := &pb.CertGen{
			Id:         uuid.NewString(),
			Issuer:     uuid.NewString(),
			Name:       uuid.NewString(),
			Number:     uuid.NewString(),
			SealName:   uuid.NewString(),
			SealNumber: uuid.NewString(),
			SignHash:   uuid.NewString(),
			Date:       time.Now().Format(time.DateTime),
			Base:       base,
		}

		genBytes, _ := sonic.Marshal(gen)
		sonic.Unmarshal(genBytes, gen)
	}
}

// 使用标准库对json进行序列化和反序列化
func genJsonStd(num int) {
	for i := 0; i < num; i++ {
		data := map[string]interface{}{
			"name":  uuid.NewString(),
			"age":   rand.Int(),
			"score": rand.Float64(),
		}
		extra, _ := structpb.NewStruct(data)

		base := &pb.Base{
			TxHash:      uuid.NewString(),
			Timestamp:   time.Now().Unix(),
			Extra:       extra,
			BlockNumber: rand.Uint64(),
			Category:    rand.Int31(),
		}

		gen := &pb.CertGen{
			Id:         uuid.NewString(),
			Issuer:     uuid.NewString(),
			Name:       uuid.NewString(),
			Number:     uuid.NewString(),
			SealName:   uuid.NewString(),
			SealNumber: uuid.NewString(),
			SignHash:   uuid.NewString(),
			Date:       time.Now().Format(time.DateTime),
			Base:       base,
		}

		genBytes, _ := json.Marshal(gen)
		json.Unmarshal(genBytes, gen)
	}
}

bench_test.go内容如下:

package main

import (
	"testing"
)

func BenchmarkGenProto(b *testing.B) {
	genProto(b.N)
}

func BenchmarkGenJsonSonic(b *testing.B) {
	genJsonSonic(b.N)
}

func BenchmarkGenJsonStd(b *testing.B) {
	genJsonStd(b.N)
}

孟斯特

声明:本作品采用署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0)进行许可,使用时请注明出处。
Author: mengbin
blog: mengbin
Github: mengbin92
cnblogs: 恋水无意


标签:rand,Protobuf,proto,JSON,vs,NewString,uuid
From: https://www.cnblogs.com/lianshuiwuyi/p/17626558.html

相关文章

  • How to use Javascript JSON.stringify similar method in Python All In One
    HowtouseJavascriptJSON.stringifysimilarmethodinPythonAllInOne如何在Python中使用类似JavaScriptJSON.stringify的方法应用场景比较两个数组(列表)对象是否相等/comparestwoarray(list)objectsforequality//jsarr1=[1,2,3]arr2=[1,2,3]......
  • 文心一言 VS 讯飞星火 VS chatgpt (75)-- 算法导论7.2 4题
    四、如果用go语言,银行一般会按照交易时间来记录某一账户的交易情况。但是,很多人却喜欢收到的银行对账单是按照支票号码的顺序来排列的。这是因为,人们通常都是按照支票号码的顺序来开出支票的,而商人也通常都是根据支票编号的顺序兑付支票。这一问题是将按交易时间排序的序列转换成按......
  • 文心一言 VS 讯飞星火 VS chatgpt (75)-- 算法导论7.2 4题
    四、如果用go语言,银行一般会按照交易时间来记录某一账户的交易情况。但是,很多人却喜欢收到的银行对账单是按照支票号码的顺序来排列的。这是因为,人们通常都是按照支票号码的顺序来开出支票的,而商人也通常都是根据支票编号的顺序兑付支票。这一问题是将按交易时间排序的序列转换成......
  • Extended Kalman Filter vs. Error State Kalman Filter for Aircraft Attitude Estim
    EKF与ESKF的对比“Engineerscansolveexactproblemsusingnumericalapproximations,ortheycansolveapproximateproblemsexactly"-FredDaum.对出现在实际问题中的非线性的运动学(dynamic)模型以及/或非线性的观测方程进行线性化的操作,然后基于这个线性化的方程计算......
  • json解析源码学习
    c#的几个Json库MiniJSONSimpleJsonlitjsonNewtonJson 其中MiniJSON最简单,所以这边也是学习这个库的Json解析部分(注意:只涉及解析,没有生成json)。整体代码也没有用到特别的算法什么的,就是一个一个字符的读取,然后根据读到的边界标识符来进行后续的读取,边界标识符的话就是:{}、[......
  • vscode取消json文件注释下划线
    使用vscode打开一个json文件,如果有单行或多行注释,则会显示红色下划线,解决办法如下:方法1点击底部的JSON,选择JSONwithComments即可,然后红色下划线消失,底部显示如下方法2方法1重启vscode可能会失效,因此在设置中搜索files.associations,然后如下图添加项这样就不会失效了!......
  • windws系统上,vscode中怎么把gin项目生产linux可执行文件
    在Windows系统上,您可以通过以下步骤在VSCode中将Gin项目生成Linux可执行文件:1.确保您已经安装了Golang和VSCode。2.打开VSCode并导入您的Gin项目。3.在终端中导航到项目的根目录。4.运行以下命令以生成Linux可执行文件:SETGOOS=linuxSETGOARCH=amd64gobuild-......
  • 制作catvsdog_path_dataset.tfrecords的代码 数据集制作完成路径为: E:\catanddog\t
    #-*-coding:utf-8-*--##PROJECT_NAME:081200#Name:01#Author:GG#Date:2023/8/12importtensorflowastfimportosimportnumpyasnpimportcv2file_dir="E:\\catanddog\\train0"save_dir="E:\\catanddog\\train1"images=[]#每张图片的路径......
  • 【JavaScript40】jquery发送jsonp
    jquery中也提供了jsonp请求服务器端fromflaskimportFlask,render_template,request,make_responseapp=Flask(__name__)@app.route("/")deffunc0():news="这是一个完整的html页面"returnrender_template("index.html",......
  • VSCode远程开发树莓派
    一、安装Remote-SSH在插件中直接搜索ssh,可找到Remote-SSH,点击安装。二、配置Remote-SSH2.1配置远程树莓派的IP和登录用户名2.2配置使用账号密码方式登录文件--->首选项--->设置--->扩展--->找到ShowLoginTerminal并勾选。三、连接树莓派可以选择在本窗口或者新建......