GJM : FlatBuffers 与 protobuf 性能比较 [转载 ]
原帖地址:http://blog.csdn.net/menggucaoyuan/article/details/34409433
原作者:企鹅 menggucaoyuan
未经原作者同意不允许转载
FlatBuffers发布时,顺便也公布了它的性能数据,具体数据请见Benchmark。
它的测试用例由以下数据构成"a set of about 10 objects containing an array, 4 strings, and a large variety of int/float scalar values of all sizes, meant to be representative of game data, e.g. a scene format."
我感觉这样测试如同儿戏,便自己设计了一个测试用例,主要关注CPU计算时间和内存空间占用两个指标,参考对象是protobuf。
测试用例为:序列化一个通讯录personal_info_list(table),通讯录可以认为是有每个人的信息(personal_info)的集合。每个人信息personal_info(table)有:个人id(uint)、名字(string)、年龄(byte)、性别(enum, byte)和电话号码(ulong)。本来我想用struct表示personal_info(table),但是struct不允许有数组或string成员,无奈我用table描述它了。相应的idl文件如下:
- //////////////////////////////////////////////////////
- //// FILE : tellist.fbs
- //// DESC : basic message for msg-center
- //// AUTHOR : v 0.1 written by Alex Stocks on June 22, 2014
- //// LICENCE :
- //// MOD :
- ////////////////////////////////////////////////////////
- namespace as.tellist;
- enum GENDER_TYPE : byte
- {
- MALE = 0,
- FEMALE = 1,
- OTHER = 2
- }
- table personal_info
- {
- id : uint;
- name : string;
- age : byte;
- gender : GENDER_TYPE;
- phone_num : ulong;
- }
- table personal_info_list
- {
- info : [personal_info];
- }
- root_type personal_info_list;
因为要以protobuf做性能参考,列出protobuf的idl文件如下:
- //////////////////////////////////////////////////////
- //// FILE : tellist.proto
- //// DESC : basic message for msg-center
- //// AUTHOR : v 0.1 written by Alex Stocks on June 22, 2014
- //// LICENCE :
- //// MOD :
- ////////////////////////////////////////////////////////
- package as.tellist;
- enum gender_type
- {
- MALE = 0;
- FEMALE = 1;
- OTHER = 2;
- }
- message personal_info
- {
- optional uint32 id = 1;
- optional string name = 2;
- optional uint32 age = 3;
- optional gender_type gender = 4;
- optional uint64 phone_num = 5;
- }
- message personal_info_list
- {
- repeated personal_info info = 1;
- }
若用C的struct描述对应的头文件(其对应的程序称之为“二进制”),如下:
- /**
- * FILE : tellist.h
- * DESC : to test tellist
- * AUTHOR : v1.0 written by Alex Stocks
- * DATE : on June 28, 2014
- * LICENCE : GPL 2.0
- * MOD :
- **/
- #ifndef __TELLIST_H__
- #define __TELLIST_H__
- enum
- {
- GENDER_TYPE_MALE = 0,
- GENDER_TYPE_FEMALE = 1,
- GENDER_TYPE_OTHER = 2,
- };
- inline const char **EnumNamesGENDER_TYPE()
- {
- static const char *names[] = { "MALE", "FEMALE", "OTHER"};
- return names;
- }
- inline const char *EnumNameGENDER_TYPE(int e)
- {
- return EnumNamesGENDER_TYPE()[e];
- }
- typedef struct personal_info_tag
- {
- unsigned id;
- unsigned char age;
- char gender;
- unsigned long long phone_num;
- char name[32];
- } personal_info;
- typedef struct personal_info_list_tag
- {
- int size;
- personal_info info[0];
- } personal_info_list;
- #endif
- // the end of the header file tellist.h
测试时,在内存中构造37个personal_info对象,并序列化之,重复这个过程100万次,然后再进行反序列化,再重复100万次。
测试结果如下(补充:tellist_pb是protobuf测试程序,tellist_fb是FlatBuffers测试程序,tellist_fb是二进制测试程序,):
- 测试环境:12Core Intel(R) Xeon(R) CPU E5-2620 0 @ 2.00GHz
- free
- total used free shared buffers cached
- Mem: 66081944 62222028 3859916 0 196448 43690828
- -/+ buffers/cache: 18334752 47747192
- Swap: 975864 855380 120484
- protobuf三次测试结果:
- bin/tellist_pb
- encode: loop = 1000000, time diff = 14210ms
- decode: loop = 1000000, time diff = 11185ms
- buf size:841
- bin/tellist_pb
- encode: loop = 1000000, time diff = 14100ms
- decode: loop = 1000000, time diff = 11234ms
- buf size:841
- bin/tellist_pb
- encode: loop = 1000000, time diff = 14145ms
- decode: loop = 1000000, time diff = 11237ms
- buf size:841
- 序列化后占用内存空间841Byte,encode平均运算时间42455ms / 3 = 14151.7ms,decode平均计算时间33656ms / 3 = 11218.7ms
- flatbuffers三次测试结果:
- bin/tellist_fb
- encode: loop = 1000000, time diff = 11666ms
- decode: loop = 1000000, time diff = 1141ms
- buf size:1712
- bin/tellist_fb
- encode: loop = 1000000, time diff = 11539ms
- decode: loop = 1000000, time diff = 1200ms
- buf size:1712
- bin/tellist_fb
- encode: loop = 1000000, time diff = 11737ms
- decode: loop = 1000000, time diff = 1141ms
- buf size:1712
- 序列化后占用内存空间1712Byte,encode平均运算时间34942ms / 3 = 11647.3ms,decode平均计算时间3482ms / 3 = 1160.7ms
- 二进制三次测试结果:
- bin/tellist
- encode: loop = 1000000, time diff = 4967ms
- decode: loop = 1000000, time diff = 688ms
- buf size:304
- bin/tellist
- encode: loop = 1000000, time diff = 4971ms
- decode: loop = 1000000, time diff = 687ms
- buf size:304
- bin/tellist
- encode: loop = 1000000, time diff = 4966ms
- decode: loop = 1000000, time diff = 686ms
- buf size:304
- 序列化后占用内存空间304Byte,encode平均运算时间14904ms / 3 = 4968ms,decode平均计算时间2061ms / 3 = 687ms
- 测试环境:1 Core Intel(R) Core(TM) i5-3210M CPU @ 2.50GHz
- free
- total used free shared buffers cached
- Mem: 753932 356036 397896 0 50484 224848
- -/+ buffers/cache: 80704 673228
- Swap: 1324028 344 1323684
- protobuf三次测试结果:
- ./bin/tellist_pb
- encode: loop = 1000000, time diff = 12451ms
- decode: loop = 1000000, time diff = 9662ms
- buf size:841
- ./bin/tellist_pb
- encode: loop = 1000000, time diff = 12545ms
- decode: loop = 1000000, time diff = 9840ms
- buf size:841
- ./bin/tellist_pb
- encode: loop = 1000000, time diff = 12554ms
- decode: loop = 1000000, time diff = 10460ms
- buf size:841
- 序列化后占用内存空间841Byte,encode平均运算时间37550ms / 3 = 12516.7ms,decode平均计算时间29962ms / 3 = 9987.3ms
- flatbuffers三次测试结果:
- bin/tellist_fb
- encode: loop = 1000000, time diff = 9640ms
- decode: loop = 1000000, time diff = 1164ms
- buf size:1712
- bin/tellist_fb
- encode: loop = 1000000, time diff = 9595ms
- decode: loop = 1000000, time diff = 1170ms
- buf size:1712
- bin/tellist_fb
- encode: loop = 1000000, time diff = 9570ms
- decode: loop = 1000000, time diff = 1172ms
- buf size:1712
- 序列化后占用内存空间1712Byte,encode平均运算时间28805ms / 3 = 9345ms,decode平均计算时间3506ms / 3 = 1168.7ms
- 二进制三次测试结果:
- bin/tellist
- encode: loop = 1000000, time diff = 4194ms
- decode: loop = 1000000, time diff = 538ms
- buf size:304
- bin/tellist
- encode: loop = 1000000, time diff = 4387ms
- decode: loop = 1000000, time diff = 544ms
- buf size:304
- bin/tellist
- encode: loop = 1000000, time diff = 4181ms
- decode: loop = 1000000, time diff = 533ms
- buf size:304
- 序列化后占用内存空间304Byte,encode平均运算时间12762ms / 3 = 4254ms,decode平均计算时间1615ms / 3 = 538.3ms
上面的二进制程序的结果无论在内存空间占用还是cpu计算时间这两个指标上都是最快的。但本文只讨论FlatBuffers和protobuf,所以不让它的结果参与比较。
从以上数据看出,在内存空间占用这个指标上,FlatBuffers占用的内存空间比protobuf多了两倍。序列化时二者的cpu计算时间FB比PB快了3000ms左右,反序列化时二者的cpu计算时间FB比PB快了9000ms左右。FB在计算时间上占优势,而PB则在内存空间上占优(相比FB,这也正是它计算时间比较慢的原因)。
上面的测试环境是在公司的Linux server端和我自己的mac pro分别进行的。请手机端开发者自己也在手机端进行下测试, 应该能得到类似的结果。Google宣称FB适合游戏开发是有道理的,如果在乎计算时间我想它也适用于后台开发。
另外,FB大量使用了C++11的语法,其从idl生成的代码接口不如protubuf友好。不过相比使用protobuf时的一堆头文件和占18M之多的lib库,FlatBuffers仅仅一个"flatbuffers/flatbuffers.h"就足够了。
GJM : FlatBuffers 与 protobuf 性能比较 [转载 ]的更多相关文章
- FlatBuffers与protobuf性能比较
FlatBuffers发布时,顺便也公布了它的性能数据,具体数据请见Benchmark. 它的测试用例由以下数据构成"a set of about 10 objects containing ...
- FlatBuffers与protobuf性能比較
FlatBuffers发布时.顺便也发布了它的性能数据,详细数据请见Benchmark. 它的測试用例由下面数据构成"a set of about 10 objects containing ...
- Graphic32中TBitmap32.TextOut性能分析[转载]
转载:http://blog.csdn.net/avan_lau/article/details/6958497 最近在分析软件中画线效率问题,发现在画一些标志性符号的方法,存在瓶颈,占用较大的时间. ...
- Ehcache/Redis/Tair缓存性能对比[转载]
FW : http://xulingbo.net/?p=434 版权归许令波所有.此处只是一个转载. 后面介绍的不同方式都有测试数据,这些测试数据都是在同一的测试环境下得出的测试结果: 测试机器的配置 ...
- 深入浅出-iOS程序性能优化 (转载)
iOS应用是非常注重用户体验的,不光是要求界面设计合理美观,也要求各种UI的反应灵敏,我相信大家对那种一拖就卡卡卡的 TableView 应用没什么好印象. iOS应用是非常注重用户体验的,不光是要求 ...
- 借助 MySQLTuner 优化 MySQL 性能(转载的一篇文章)
MySQLTuner 是一个 Perl 脚本,可以用来分析您的 MySQL 性能,并且基于收集到的信息给出相应的优化建议.这样子,您就可以调整 my.cnf 从而优化您的 MySQL 设置. 这边只是 ...
- Tomcat 6 支持 NIO -- Tomcat的四种基于HTTP协议的Connector性能比较(转载)
Tomcat从5.5版本开始,支持以下四种Connector的配置分别为: <Connector port="8081" protocol="org.apache. ...
- Tomcat性能优化(转载)
出处:微信订阅号GitChat精品课程 — Tomcat性能优化 Tomcat 简单介绍 Sun 公司创建了第一个 Servlet 容器,即 Java Web Server,但 JWS 只是为了演示 ...
- Android_性能优化转载
胡凯 RSS Blog Archives Android Training in Chinese About Android性能优化典范 JAN 17TH, 2015 | COMMENTS 2015年 ...
随机推荐
- git忽略以点开头的文件夹
git忽略以点开头的文件夹 好像不是什么问题,可是我用的时候不好使,还是记录下 参考:http://www.oschina.net/question/1437985_2181276
- 字符串中判断存在的几种模式和效率(string.contains、string.IndexOf、Regex.Match)
通常情况下,我们判断一个字符串中是否存在某值常常会用string.contains,其实判断一个字符串中存在某值的方法有很多种,最常用的就是前述所说的string.contains,相对来说比较常用的 ...
- Java-集合=第五题 (Map)设计Account 对象如下: private long id; private double balance; private String password; 要求完善设计,使得该Account 对象能够自动分配id。 给定一个List 如下: List list = new ArrayList(); list.add(new A
第五题 (Map)设计Account 对象如下: private long id; private double balance; private String password; 要求完善设计,使得 ...
- 【VS技巧】根据XML自动生成类型
.NET 4.5对应的VS版本(不要问我哪个版本)中新增了一个功能,严重实用,可以根据XML文档生成新类型.这个功能在VS的[编辑]>[选择性粘贴]菜单中.怎么玩?不急,咱们实际操作一下. 以网 ...
- .NET面试题解析(05)-常量、字段、属性、特性与委托
系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 弱小和无知不是生存的障碍,傲慢才是!——<三体> 常见面试题目: 1. const和reado ...
- Android入门(十二)SQLite事务、升级数据库
原文链接:http://www.orlion.ga/610/ 一.事务 SQLite支持事务,看一下Android如何使用事务:比如 Book表中的数据都已经很老了,现在准备全部废弃掉替换成新数据,可 ...
- NYOJ995硬币找零(简单dp)
/* 题意:给你不同面额的硬币(每种硬币无限多),需要找零的面值是T,用这些硬币进行找零, 如果T恰好能被找零,输出最少需要的硬币的数目!否则请输出剩下钱数最少的找零方案中的最少硬币数! 思路:转换成 ...
- 构建自己的PHP框架--搭建基本结构
首先,我们来说一下,为什么要创建自己的框架? 为什么要创建自己的框架呢?如果你跟周围的人讨论,每个人都会告诉你重复发明轮子是一件糟糕的事情,你最好选择一个已有的框架,忘掉“创建自己的框架”这种想法.大 ...
- c#用socket异步传输字符串
再次特别感谢张子阳老师的文章,是我深感益处. 在前一篇文章中可以看到,尽管消息分成了三条单独发送,但是服务端却将后两条合并成了一条.对于这些情况,我们可以这样处理:就好像HTTP协议一样,在实际的请求 ...
- andriod + @的区别
Android中的组件需要用一个int类型的值来表示,这个值也就是组件标签中的id属性值. id属性只能接受资源类型的值,也就是必须以@开头的值,例如,@id/abc.@+id/xyz等. 如果在@后 ...