C#中使用ProtoBuf提高序列化速度对比二进制序列化

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C#中使用ProtoBuf提高序列化速度对比二进制序列化相关的知识,希望对你有一定的参考价值。


场景

ProtoBuf

protocolbuffer是google 的一种数据交换的格式,它独立于语言,独立于平台。
google 提供了多种语言的实现:java、c#、c++、go 和 python,每一种实现都包含了相应语言的编译器以及库文件。
由于它是一种二进制的格式,比使用xml 进行数据交换快许多。可以把它用于分布式应用之间的数据通信或者异构环境下的数据交换。
作为一种效率和兼容性都很优秀的二进制数据传输格式,可以用于诸如网络传输、配置文件、数据存储等诸多领域。

.proto

类似于.json和.xml,ProtoBuf有自己的文件格式.proto文件格式。

也有自己的语法,具体可以搜索.proto语法。


实现

新建.proto文件

这里使用的是EditPlus新建txt文件,按照其语法要求编写如下request.proto文件。

package ProtoBufTest;

message Request
required int32 id = 1;
required string password = 2;

注:

package ProtoBufTest 要与项目的namespace相对应。

message 后面就是要生成的类名。

required表示必须的,int32对应int类型,string对应string类型。

=1是固定的语法格式,记得往后递增。

最终文件如下

C#中使用ProtoBuf提高序列化速度对比二进制序列化_System

 

.ptoto文件生成类(.cs文件)

将上面新建的文件放在与protogen.exe同目录下

C#中使用ProtoBuf提高序列化速度对比二进制序列化_System_02

 

然后在此处打开命令行(Windows下是按住shift,在当前目录右击选择在此处打开命令行)。

输入命令:

protogen.exe   -i:request.proto    -o:Request.cs

C#中使用ProtoBuf提高序列化速度对比二进制序列化_ProtoBuf_03

 

注:

-i后面跟的是上面新建的proto文件

-o后面跟的是要生成的.cs文件

运行后会在同目录下生成Request.cs

C#中使用ProtoBuf提高序列化速度对比二进制序列化_数据交换_04

 

生成的文件内容

//------------------------------------------------------------------------------
// <auto-generated>
// This code was generated by a tool.
//
// Changes to this file may cause incorrect behavior and will be lost if
// the code is regenerated.
// </auto-generated>
//------------------------------------------------------------------------------

// Generated from: request.proto
namespace ProtoBufTest

[global::System.Serializable, global::ProtoBuf.ProtoContract(Name=@"Request")]
public partial class Request : global::ProtoBuf.IExtensible

public Request()

private int _id;
[global::ProtoBuf.ProtoMember(1, IsRequired = true, Name=@"id", DataFormat = global::ProtoBuf.DataFormat.TwosComplement)]
public int id

get return _id;
set _id = value;

private string _password;
[global::ProtoBuf.ProtoMember(2, IsRequired = true, Name=@"password", DataFormat = global::ProtoBuf.DataFormat.Default)]
public string password

get return _password;
set _password = value;

private global::ProtoBuf.IExtension extensionObject;
global::ProtoBuf.IExtension global::ProtoBuf.IExtensible.GetExtensionObject(bool createIfMissing)
return global::ProtoBuf.Extensible.GetExtensionObject(ref extensionObject, createIfMissing);


项目中引进

新建窗体项目,然后将上面生成的Request.cs文件复制到项目下。

此时打开cs文件会报错,此时需要引进dll(动态链接库文件)--protobuf-net.dll。


添加引用

打开资源管理器--右击引用--添加

C#中使用ProtoBuf提高序列化速度对比二进制序列化_数据交换_05

 

选择浏览--右下角浏览--确定

C#中使用ProtoBuf提高序列化速度对比二进制序列化_序列化_06

 

实现ProtoBuf序列化

打开窗体设计器,拖拽一个按钮Button,然后双击按钮进入其点击事件。

private void button1_Click(object sender, EventArgs e)

//序列化操作
Request request = new Request();
request.id = 1;
request.password = "123";
//计时器计时
Stopwatch sw = new Stopwatch();
//启动计时器
sw.Start();
//执行序列化
MemoryStream ms = new MemoryStream();
Serializer.Serialize<Request>(ms, request);
byte[] data = ms.ToArray();
//停止计时器
sw.Stop();
//输出计时时间
Console.WriteLine("使用ProtoBuf序列化:" + sw.Elapsed);

注:

在上面调用Serializer时引入的是ProtoBuf自带的。

C#中使用ProtoBuf提高序列化速度对比二进制序列化_ProtoBuf_07

 

对比二进制序列化

右击项目-添加-类

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ProtoBufTest

[Serializable]
class Person


public Person(int id ,string password)
this.Id = id;
this.Password = password;

private int id;

public int Id

get return id;
set id = value;


private string password;

public string Password

get return password;
set password = value;


注:记得要添加可序列化的标识[Serializable]

在上面的窗体中在拖拽一个Button,双击进入其点击事件。

private void button2_Click(object sender, EventArgs e)

Person per = new Person(18,"张三");
//计时器计时
Stopwatch sw = new Stopwatch();
sw.Start();
FileStream fs = new FileStream(filePath, FileMode.Create);
BinaryFormatter bf = new BinaryFormatter();
bf.Serialize(fs, per);
fs.Close();
sw.Stop();
Console.WriteLine("二进制序列化:" + sw.Elapsed);

对比效果

运行项目,先点击二进制序列化按钮,再点击ProtoBuf序列化按钮。

 

C#中使用ProtoBuf提高序列化速度对比二进制序列化_数据交换_08

 

以上是关于C#中使用ProtoBuf提高序列化速度对比二进制序列化的主要内容,如果未能解决你的问题,请参考以下文章

google protobuf 数据类型_理解Protobuf数据格式解析

protobuf使用简介

Protobuf 消息是不是独立于平台

ProtoBuf简介

Google Protobuf简明教程

Google Protobuf简明教程