LiteByte教程

简介

LiteByte是一种轻量级的二进制数据交换格式。
体积小巧、简单易用是设计目标。主要用于解决前后台数据传输量的问题。

作者:冰封百度(ZhangYu)
设计的灵感来源于C# struct内存对齐后的紧凑格式。暂时只实现了C#版本。

Gitee:https://gitee.com/zhangyu800/litebyte

特点

1.紧凑的二进制数据格式,支持变长整型,数据量小。
2.用类定义对象结构,使用方便。

实现思路

把一个对象分为两个部分:结构和值。

结构用类定义,越方便越好。

值用于网络传输,越小越好。

 

             // 比如像这样的对象 拆分成结构和值
             public class UserInfo {

                 public int id = 1001;
                 public string name = "冰封百度";

             }

//              ↙              ↘
// 结构                           //
public class SUserInfo {       

    public int id;                // 1001        = [0x02, 0xE9, 0x03]
    public string name;           // "冰封百度"   = [0x00, 0x05, 0xB0, 0x51, 0x01, 0x5C, 0x7E, 0x76, 0xA6, 0x5E]

}

// 对象的值就是:[0x02, 0xE9, 0x03, 0x05, 0xB0, 0x51, 0x01, 0x5C, 0x7E, 0x76, 0xA6, 0x5E]

 

前后台依赖相同的结构类,转换时把对象的值拆出来传输,解析时把值还原成对象。

测试Demo:

https://pan.baidu.com/s/18fbL0uG01gKW24KgtfR2bw?pwd=hhqu

 

使用方法

1.创建自定义结构类(UserInfo)。
2.添加用[LBMember]标记要序列化的字段/属性。
3.调用LBUtil.Serialize(instance) 把对象序列化成二进制数据。
4.调用LBUtil.Deserilize<T>(bytes) 把二进制数据反序列化成对象。

代码样例:

// 1.自定义对象结构类:
// 2.using LiteByte
// 3.用LBMember标记要序列化的字段或属性

using System;
using LiteByte;
using System.Collections.Generic;

/// <summary> 玩家信息测试 | PlayerInfo test </summary>
public class STPlayerInfo {

    [LBMember(0)]
    public uint id;
    [LBMember(1)]
    public string nickname;
    [LBMember(2)]
    public byte gender;
    [LBMember(3)]
    public bool isVip;
    [LBMember(4, LBTypeCode.VarInt32)]
    public int lv;
    [LBMember(5)]
    public int hp;
    [LBMember(6)]
    public int mp;
    [LBMember(7, LBTypeCode.VarUnicode)]
    public string desc;
    [LBMember(8)]
    public List<int> ids;
    [LBMember(9)]
    public Dictionary<int, string> dic;
    [LBMember(10, LBTypeCode.ASCII, LBTypeCode.VarUnicode)]
    public Dictionary<string, string> dic2;

}

// 序列化和反序列化:
public void PlayerInfoTest() {
    Dictionary<int, string> dic = new Dictionary<int, string>();
    dic.Add(1, "AAA");
    dic.Add(2, "BBB");

    Dictionary<string, string> dic2 = new Dictionary<string, string>();
    dic2.Add("A", "ValueA");
    dic2.Add("B", "ValueB");

    STPlayerInfo st = new STPlayerInfo();
    st.id = 100001;
    st.nickname = "冰封百度";
    st.gender = 1;
    st.isVip = true;
    st.lv = 999;
    st.hp = 999999;
    st.mp = 999999;
    st.desc = "这是一个Demo";
    st.ids = new List<int>() { 1, 2, 3, 4, 5 };
    st.dic = dic;
    st.dic2 = dic2;

    // 序列化:
    byte[] bytes = LBUtil.Serialize(typeName, player);

    // 反序列化:
    PlayerInfo info = LBUtil.Deserialize<PlayerInfo>(typeName, bytes);
    // bytes长度:81字节
}

 

转换结果:

 

代码说明:

1.序列化对象:LBUtil.Serialize(obj)
2.反序列化对象:LBUtil.Deserilize<T>(bytes)
3.可以转换自定义的class和struct 字段(属性)需要是public的

支持的数据类型

数据类型介绍

1.基本的值类型:bool、byte、short、int、long、float、double等
2.字符串 string (支持ASCII、Unicode、UTF8、VarUnicode四种编码方式)
3.数组和List(int[] 或 List<int>这种)
4.字典(Dictionary<K,V>)
4.自定义类型(class或struct)

数据类型说明:

1.对bool型的支持最好,一个bool型只占1位(1/8个字节)。
2.支持变长整数(short、int、long、ushort、uint、ulong)
3.支持null值 (能空的类型string, array, object,都支持它们为空的情况)
4.建议在定义数据格式时,用尽量小的类型定义字段,这样序列化的数据体积会更小,如果懒得写,可以考虑使用变长数据。
5.支持自定义字符串编码格式 每个字段都可以自定义。
6.因为在编写变长数据类型的过程中用到了一些不常见的数据格式,为了重用类型,索性就一起支持了,在对数据大小很严格的环境会有帮助。

基本数据类型

比特型(7种)

 

类型 长度 值范围
Bit1(Boolean) 1位 0 ~ 1
Bit2(Byte) 2位 0 ~ 3
Bit3(Byte) 3位 0 ~ 7
Bit4(Byte) 4位 0 ~ 15
Bit5(Byte) 5位 0 ~ 31
Bit6(Byte) 6位 0 ~ 63
Bit7(Byte) 7位 0 ~ 127

 

整型(16种)

类型 长度  值范围
Int8(sbyte) 1字节 -128 ~ 127
Int16(short) 2字节 -32768 ~ -32767
Int24(int) 3字节 -8388608 ~ 8388607
Int32(int) 4字节 -2147483648 ~ 2147483647
Int40(long) 5字节 -549755813888 ~ 549755813887
Int40(long) 6字节 -140737488355328 ~ 140737488355327
Int40(long) 7字节 -36028797018963968 ~ 36028797018963967
Int64(long) 8字节 -9223372036854775808 ~ 9223372036854775807
UInt8(byte) 1字节 0 ~ 255
UInt16(ushort) 1字节 0 ~ 65535
UInt24(uint) 1字节 0 ~ 16777215
UInt32(uint) 1字节 0 ~ 4294967295
UInt40(ulong) 1字节 0 ~ 1099511627775
UInt48(ulong) 1字节 0 ~ 281474976710655
UInt56(ulong) 1字节 0 ~ 72057594037927935
UInt64(ulong) 1字节 0 ~ 18446744073709551615

 

浮点型(5种)

类型 长度 有效数字 值范围
Float8(float) 1字节 7位 0/255 ~ 255/255
Float16(float) 2字节 3位 ±6.55E +4
Float24(float) 3字节 5位 ±1.8447E +19
Float32(float) 4字节 7位 ±3.402823E +38
Float64(double) 8字节 15位 ±1.7976931348623157E +308

 

变长整型(7种)

类型 长度 值范围
VarInt16(short) 1位 + 1~2字节 同Int16
VarInt32(int) 2位 + 1~4字节 同Int32
VarInt64(long) 3位 + 1~8字节 同Int64
VarUInt16(ushort) 1位 + 1~2字节 同UInt16
VarUInt32(uint)  2位 + 1~4字节 同UInt32
VarUInt64(ulong) 3位 + 1~8字节 同UInt64
VarLength(int)  3位 + 1~8字节 -1 ~ (Int32.MaxValue/2 - 1)

 

字符串(4种编码)

类型 单个字符长度 总长度范围
ASCII(string) 1字节 头(1~4)字节+体(0 ~ 1073741822)字节
Unicode(string) 2字节 头(1~4)字节+体(0 ~ 1073741822)x2字节
UTF8(string) 1~4字节 头(1~4)字节+体(0 ~ 1073741822)字节
VarUnicode(string) 1~4字节 同VarLength + VarUnicode

 

复杂数据类型(4种)

类型 表达式
数组 T[]
列表 List<T>
字典 Dictionary<K, V>
自定义类型 只要不和基本类型和数组重名 即被当作自定义类型

 

更新日志:

0.7.0: 用IL.Emit代替反射,大幅度提升get和set性能 IL优化后性能比直接=稍慢大概一倍,优化之前反射比直接=慢几十倍。
0.8.0: 添加编译多个类的功能,一个配置文件可以写多个类了,可以一次性识别。
0.8.1:删除LBObject类,因为IL优化后get和set的性能很好,不再需要LBObject了。
0.8.2:优化LBParser中 从tokens中删除访问修饰符的代码,不再生成新的list。
0.8.3:优化反射工具类 ReflectionUtil 创建实例方法中的参数
0.9.0: 优化反射工具(增加IL.Emit工具、委托工具、指针工具 用于优化反射) 增加反射工具对IL2CPP模式的兼容性
0.10.0: 重构LBReader读取类 增加对List的支持 优化LBConverter类
0.10.1: 修复BUG:1.LBWriter WriteVarLength时没有正确扩容的BUG 2.LBConverter没有正确抛出错误信息的BUG
0.11.0: 重构LBWriter写入类 增加对List的支持 优化LBConverter类
0.11.1: 优化LBConverter.ToObject() 和 ToBytes()
0.12.0:重构序列化/反序列化方式 不再使用配置文件 直接使用类+属性标签的方式完成,删除读配置文件的所有功能。增加对List<T>和Dictionary<K,V>的支持。

其他说明:

由于能力有限,暂时只实现了C#版本(在Unity中实现的,算半个.Net吧)
其他语言后续有时间再写,虽然造了个轮子 不过感觉造轮子的过程中收获远大于付出,挺开心的。
建了个群,有需求的可加。
QQ群:715800513

项目Gitee:https://gitee.com/zhangyu800/litebyte

测试Demo:
链接:https://pan.baidu.com/s/18fbL0uG01gKW24KgtfR2bw?pwd=hhqu

转载请标明原文地址:https://segmentfault.com/a/1190000021329368

 

posted @ 2019-12-20 14:47  冰封百度  阅读(534)  评论(0编辑  收藏  举报