ProtoBuf在java中的实际应用

在实际的应用之前,我们再了解以下protobuf。
通过比较它与其他数据格式进行比较,可以使我们更好的认识它的应用场景。下面与XML,JSON进行一个简单的比较。

  • JSON:一般在web项目中广泛使用,主要是由于浏览器的支持非常好,内部构建了与多函数来支持JSON。具有可读性。
  • XML:在WebService中广泛使用,但是过于冗余(毕竟是通过标签进行标识)。也具有可读性。
  • ProtoBuf:适合高性能,对响应速度有要求的数据传输场景。由于是二进制数据格式需要编码和解码。不具有可读性。在官网上描述ProtoBuf比XML小3到10倍,快20到100倍。

结论: 在一个需要大量的数据传输的场景中,如果数据量很大,那么选择protobuf可以明显的减少数据量,减少网络IO,从而减少网络传输所消耗的时间。


在上面我们对ProtoBuf与其他数据格式做了比较,只简单的了解到它的小和快,可这是如何实现的呢?那么我们在下面对ProtoBuf存储原理做一个简要的介绍。以下内容转载自博文-google protobuf存储原理及一些底层api应用

核心是Google 提出了“Base 128 Varints”编码,这是一种变字节长度的编码,官方描述为:varints是用一个或多个字节序列化整形的一种方法。

序列化方式
protobuf 把 message通过一系列key_value对来表示。
Key 的算法为:(field_number << 3)| wired_type
这里field_number 就是具体的索引,wired_type的值按下表查询。

wired_type .proto类型
0 Varint int32, int64, uint32, uint64, sint32, sint64, bool, enum
1 64-bit fixed64, sfixed64, double
2 Length-delimited string, bytes, embedded messages, packed repeated fields
5 32-bit fixed32, sfixed32, float

对于int,bool,enum类型,value就是Varint。
而对于string,bytes,message等等类型,value是长度+原始内容编码。

  • 举例int类型存储(Varint存储原理)
    存储一个int32 类型的数字,通常是4个字节。但是Varints最少只需要一个字节就可以了。
    Varints规定小于128的数字都可以用一个字节来表示,比如10, 它就会用一个字节 0000 1010 来存储。
    对于大于128的数字,则用更多个字节存储。
    以150举例:protobuf的存储字节是 1001 0110 0000 0001。
    为什么会这样标识呢?首先我们了解一个字节共8位,表示的数字是255,但是Varints只用一个字节表示小于128的数字,换句话说,就是Varints只用了8位中的7位来表示数字,而还有一位被用来干嘛了呢?
    Varints在官方规定中表示,每个字节的最高位是有特殊含义,当最高位为1的时候,代表后续的字节也是该数字的一部分,后续为0的时候,则表示结束。
    比如过150,二进制表示为 1001 0110。
    先取后七位 001 0110, 作为第一个字节的内容。
    再取余下的位,补0凑齐7位,就是000 0001。
    对于intel机器,是小端字节序,低字节位于地址低的。0010110 是低字节地址,因此排在前面,因为后面的也是数字的一部分,所以高位补1,也就成了10010110。 同样的,高字节000 0001,排在后面,并且它后面没有后续字节了,所以补0,也就成了 0000 0001。
    因此150 在protobuf中的表示方式为 1001 0110 0000 0001。
  • 举例string类型存储
message Test {
    required string desc = 2;
}

假如把 a 设置为 “testing”的话, 那么序列化后的就是
12 07 74 65 73 74 69 64 67
其中12是key。剩下的是value。
怎么算的呢?先看12, 这里的12,是个16进制数字,其二进制位表示为 0001 0010。
0010 就是类型string的对应的Type值,根据上表,也就是2。
field_number 是 2,也就是0010,左移三位,就成了0001 0000。
按照key的计算公式,和Type值取并后就变成了 0001 0010,即12。
Value是长度加原始内容编码。
07就是长度, 代表string总长7个字节。 后面7个数字一次代表每个字母所对应的16进制表示。


下面通过java配合protobuf进行一些简单的应用,以下操作类请参考初识ProtoBuf,添加依赖或者jar包,此处我是通过maven项目进行演示。

<!-- https://mvnrepository.com/artifact/com.google.protobuf/protobuf-java -->
<dependency>
    <groupId>com.google.protobuf</groupId>
    <artifactId>protobuf-java</artifactId>
    <version>3.6.1</version>
</dependency>

java代码展示如下:

//创建对象,及属性赋值
PersonProto.Person.Builder builder = PersonProto.Person.newBuilder();

builder.setName("Mrzhang")
        .setAge(18)
        .setSex(true)
        .setBirthday(System.currentTimeMillis())
        .setAddress("军事基地")
        .addCars(0, PersonProto.Car.newBuilder().setName("兰博基尼").setColor("Red").build())
        .putOther("描述", "暂无");
PersonProto.Person person = builder.build();

//序列化(通过protobuf生成的java类的内部方法进行序列化)
byte[] bytes = person.toByteArray();

//反序列化(通过protobuf生成的java类的内部方法进行反序列化)
try {
        PersonProto.Person parseFrom = PersonProto.Person.parseFrom(bytes);
} catch (InvalidProtocolBufferException e) {
        e.printStackTrace();
}

我们可以将此对象的序列化byte[]传送给其他服务,让他们通过同一个.proto文件生成相应语言的文件,通过内部的方法进行反序列化。

举例

  • 两个相互独立的项目,但是会操作同一数据,并且数据进行缓存,可以将序列化byte[]存放在Redis
  • 不同语言通过的数据通信

ProtoBuf与Json的转换
为什么要做这样的转换呢?因为我想到与数据库对应的javabean是否可以转换为proto数据这样就可以进行数据进行后续的数据交互了。(当然也有其他方式,例如反射,和复制属性等方式)

//通过fastJson进行转换
String json = JsonUtil.toJson(person);

让人意外的是,转换失败。提示信息一大堆,大致意思就是不能够转换。
通过ProtoBuf Util转换
添加依赖

<!-- https://mvnrepository.com/artifact/com.google.protobuf/protobuf-java-util -->
<dependency>
    <groupId>com.google.protobuf</groupId>
    <artifactId>protobuf-java-util</artifactId>
    <version>3.6.1</version>
</dependency>

java代码展示如下:

//to Json
JsonFormat.Printer printer = JsonFormat.printer();
String print = "";
try {
    print = printer.print(person);
    System.out.println(print);
} catch (InvalidProtocolBufferException e) {
    e.printStackTrace();
}

//to Object
JsonFormat.Parser parser = JsonFormat.parser();
try {
    PersonProto.Person.Builder newBuilder = PersonProto.Person.newBuilder();
    parser.merge(print, newBuilder);
    System.out.println(newBuilder.build());
} catch (InvalidProtocolBufferException e) {
    e.printStackTrace();
}

//添加java bean 此类对性数据库的字段,同时与proto类属性名相同
public class Person implements Serializable {
    private String name;
    private Integer age;
    private Boolean sex;
    private Date dirthday;//此处注意这里是时间类型而非proto类中的long类型
    private String address;
    private List<Car> cars = new ArrayList<Car>();
    private Map<String, String> other = new HashMap<String, String>();
}

public class Car implements Serializable {
    private String name;
    private String color;
}

//在上面的转换中间添加以下代码,发现同样转换成功
Person myPerson = JsonUtil.toObject(print, Person.class);
System.out.println(myPerson);
print = JsonUtil.toJson(myPerson);

总结:通过上面的测试,我们发现可以实现自定义的bean与proto是可以通过Json相互转换的,然而它们之间的转换需要第三方JSON转换工具和protobuf util的支持。
上面可以应用于自定义的bean主要用于web前后台的交互,同时通过JSON转换也可以进行与其他服务的交互,数据可以放在redis中。
当然还有其他的转换方式,如我上面所说:反射,和复制属性。
反射可以通过属性名称一一对应进行转换,在特定的情况下也是可以通过属性类型进行转换。
复制属性,可以通过Spring的BeanUtils.copyProperties(Object source, Object target)方法进行转换的。
后期我将补充此转换方式的案例。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容