• 网络游戏协议:基于Protobuf的序列化与反序列化


    本节給大家讲解的是网络游戏开发中的序列化与反序列化。当我们要存储/传送一个数据对象的数据的时候,我们要把这个数据对象实例编码成二进制数据,这样就可以把这些二进制进行存储与传送。当我们接收读取这些二进制的时候,我们有可以根据数据把对应的数据对象实例重建出来,这个过程,我们叫做序列化与反序列化。本节将从以下3点来详细的讲解基于Protobuf的序列化与反序列化。如下:

    对啦!这里有个unity学习交流小组里面聚集了一帮热爱学习unity的零基础小白,也有一些正在从事unity开发的技术大佬,欢迎你来交流学习。

    基于二进制数据序列化与反序列化的基本原理详解

    我们先来通过一个简单的案例,来讲解二进制序列化与反序列化的基本原理。任何一个复杂对象的数据都可以由简单的数据组合而成,比如 Vector3 结构体对象,就是由3个float分量x, y, z组合而成。数据对象的基本数据类型包括有: Int, float, double, string, boolean等。任何对象可以内嵌子对象,或者使用数组来存储多个子对象。子对象与数组,让我们的数据对象的组成关系可以变得非常复杂

    接下来我们以一个对象为例来进行讲解:

    1. struct Vector {
    2. int x;
    3. int y;
    4. int z;
    5. }
    6. struct Size {
    7. int w;
    8. int h;
    9. }
    10. class Rect {
    11. Vector org;
    12. Size s;
    13. }

    Rect对象里面包含了Vector类与Size类子对象, 而这些子对象由基本的数据组成。我们要编码Rect对象,就要先编码Vector对象,再编码Size对象,这样才能得到Rect对象得编码。解下来我们针对Rect对象做编写编码函数,如下:

    基本数据类型得编码函数,这个不用变,可以一次性写好,后面反复得使用;

    1. byte[] Float_encode(float) {…}
    2. byte[] Int_encode(int) {…}
    3. byte[] string_encode(string) {…}
    4. byte[] bool_encode(bool) {…}

    每个复杂的数据对象的内部结构都是开发者定义的,所以这块必要更具应用有多少个对象,来编写函数实现,而这些函数可以调用基本数据类型的编码函数,最后组合成byte[];

    1. byte[] Vector_encode(Vector v) {
    2. byte[] b1 = int_encode(v.x);
    3. byte[] b2 = int_encode(v.y);
    4. byte[] b3 = int_encode(v.z);
    5. return b1 + b2 + b3;
    6. }
    7. byte[] Size_encode(Size s) {
    8. byte[] b1 = int_encode(s.w);
    9. byte[] b2 = int_encode(s.h);
    10. return b1 + b2;
    11. }
    12. byte[] Rect_encode(Rect r) {
    13. byte[] b1 = Vector_encode(r.org);
    14. byte[] b2 = Size_encode(r.s);
    15. return b1 + b2;
    16. }

    解码也类似,基本的数据类型,可以实现一次解码函数,复杂的对象,都是由开发者自己调用基本数据类型的解码,一个个的解码出来。

    经过上面的分析,我们总结如下:

    Protobuf的设计架构与对应工具

    上面总结分析出来的这四点,为了解决上面的问题,google 发起了一个protobuf的开源项目。目前我们做网络游戏中序列化、反序列化很多都是基于它来做的。Protobuf解决上序4个问题,主要的设计如下:

    (1) 将基本数据类型的二进制数据编码/解码,编写成了固定的函数,形成一个”库”,这个库我们叫做runtime库;

    (2) 我们将runtime库实现多个语言版本,支持主流的编程语言;

    (5) 开发者只需要将要序列化/反序列化的数据对象定义到协议文件,然后运行编译器生成协议文件对应的目标编程语言的代码, 就可以使用里面的类与对象了。

    Protoc协议编译器,是C/C++开发的,可以生成多个编程语言,在编译protobuf的时候,这个工具也会同时被编译出来。Runtime库代码直接拷贝过去即可。协议代码生成后,放到工程项目中直接使用。代码中直接使用这些生成的数据类型。

    Unity如何集成使用Protobuf

    Unity项目如何使用Protobuf呢?按照下面的步骤来实现可以了。

    (1) 内置C#的protobuf的runtime库代码到Unity项目,复制过来,直接放Unity就可以了。我一般会在Scripts下新建一个3rd文件夹来存放,如下图所示:

    (2) 使用cmake源码编译protobuf,来生成protoc的编译工具,如果自己懒得编译,可直接到github上下载已经编译好的protoc。

    (3) 在Unity Assets文件夹外面定义一个协议生成文件夹,专门用来定义协议文件.proto, 然后运行编译器生成 c#代码,然后再拷贝到项目Assets的指定代码目录下。

    今天的分享就到这里了,关注我,学习更多网络游戏相关的开发知识。

  • 相关阅读:
    idea2021 创建Servlet模板 & idea2021如何修改Servlet模板
    56. 合并区间
    matlab simulink 模糊变论域控制电梯四分之一模型
    公纵号发送提示信息(用户微服务--消息微服务)
    Queue 中 poll()和 remove()的区别(详解)
    pcsx2模拟器怎么设置流畅?
    网站分类seo怎么优化(如何调整有利于SEO排名)
    Python 求矩阵的局部极大值
    深入理解Redis事务、事务异常、乐观锁、管道
    关于Python Ansible中 HOST_KEY_CHECKING不生效问题分析
  • 原文地址:https://blog.csdn.net/voidinit/article/details/126279290