protobuf和protostuff的区别
在我们的开发过程中,序列化是经常需要处理的问题,比如在做分布式访问数据时,或者是在做redis缓存存储数据时,如果我们涉及的知识面不够广的话,可能会简单的使用JDK的序列化,也即在需要序列化的类上implements Serializable接口去实现序列化,我想说的是这种方式在小系统中尚且可以用一用,如果是并发很大的系统会受到严重影响,这是由于JDK自带的序列化效率很低,不论是时间上还是空间上。我们经常使用的序列化方式还有XML和Json,说实在的我更多的是使用Json,我觉得它很方便很友好,但这些都不够好,我今天要将的是google开发的开源的序列化方案protocol buffer(简称protobuf),它的好处很多,独立于语言,独立于平台,最最重要的是它的效率相当高,用protobuf序列化后的大小是json的10分之一,xml格式的20分之一,是二进制序列化的10分之一,是不是很心动。其实我也刚接触这个好东西,写下此篇博客就当一个学习笔记吧。
protobuf使用起来非常简单,它的主要流程是:我们需要自己写一个.proto文件用来描述序列化的格式,然后用protobuf提供的protoc工具将.proto文件编译成一个Java文件(protobuf官方支持很多语言:Java、C++、C#、Go、Python ,protobuf是一个开源项目,因此有很多大牛也实现了其他语言,但它们的可靠性还有待验证),最后将该Java文件引入到我们的项目中就可以使用了,当然还得引入protobuf的依赖包。
1、我们需要到官网下载protobuf的相应版本,我这里下载的是windows下的3.1.0版protoc-3.1.0-win32.zip
2、将下载好的zip解压,能看到bin目录下有一个protoc.exe的文件,等下需要用它来编译文件,我们直接在bin目录下 创建一个简单的person.proto的描述文件,内容如下:
- syntax = "proto3";
- option java_package = "gudao.red.protobuf";
- option java_outer_classname = "PersonFactory";
- message Person{
- int32 id = 1;
- string name = 2;
- int32 age = 3;
- Addr addr = 4;
- }
- message Addr{
- string contry = 1;
- string city = 2;
- }
内容非常简单,大概介绍一下:
syntax = "proto3"; 我们使用proto3版协议
option java_package = "gudao.red.protobuf"; 编译之后生成的Java文件的包名
option java_outer_classname = "PersonFactory"; 编译之后生成的Java类的类名
message 相当于Java中的class
详细的介绍,还请自行去官网查看
3、使用protoc编译上述.proto文件,生成Java类,使用如下命令完成该操作
- protoc --java_out=./src ./person.proto
--java_out:生成的Java文件输出的位置,其他语言有相应的选项
这样就会在src目录下生成一个 名为PersonFactory的Java文件
4、将PersonFactory.java文件引入到我们的项目中,并引入对应版本的protobuf的依赖包
5、写测试代码
- package gudao.red.protobuf_test;
- import java.net.Socket;
- import gudao.red.protobuf.PersonFactory.Addr;
- import gudao.red.protobuf.PersonFactory.Person;
- public class Client {
- public static void main(String[] args) throws Exception {
- Socket socket = new Socket("127.0.0.1",3030);
- Person.Builder person = Person.newBuilder();
- Addr.Builder addr = Addr.newBuilder();
- addr.setContry("china").setCity("shenzhen");
- person.setId(1).setAge(12).setName("ccf");
- person.setAddr(addr);
- byte[] messageBody = person.build().toByteArray();
- int headerLen = 1;
- byte[] message = new byte[headerLen+messageBody.length];
- message[0] = (byte)messageBody.length;
- System.arraycopy(messageBody, 0, message, 1, messageBody.length);
- System.out.println("msg len:"+message.length);
- socket.getOutputStream().write(message);
- }
- }
- package gudao.red.protobuf_test;
- import java.net.ServerSocket;
- import java.net.Socket;
- import gudao.red.protobuf.PersonFactory.Person;
- public class Server {
- /**
- * @param args
- */
- public static void main(String[] args) throws Exception {
- // TODO Auto-generated method stub
- ServerSocket serverSock = new ServerSocket(3030);
- while(true){
- Socket sock = serverSock.accept();
- byte[] msg = new byte[256];
- sock.getInputStream().read(msg);
- int msgBodyLen = msg[0];
- System.out.println("msg body len:"+msgBodyLen);
- byte[] msgbody = new byte[msgBodyLen];
- System.arraycopy(msg, 1, msgbody, 0, msgBodyLen);
- Person person = Person.parseFrom(msgbody);
- System.out.println("Receive:");
- System.out.println(person);
- }
- }
- }
先后启动Server和Client,就可以看到控制台的输出如下:
至此,我们的简单使用过程就完成了,是不是很简单。是,这个例子看上去是挺简单的,但如果我们需要序列化的类非常多,那么我们是不是得写非常多的.proto文件,并且还需要更新它们,这个代价可以想象一下也是非常大的。那么,接下来我们就来讲一讲protostuff,看这名字是不是跟protobuf很像,嗯,它们是有关系,前者就是基于后者实现的。
6、protostuff是一个基于protobuf实现的序列化方法,它较于protobuf最明显的好处是,在几乎不损耗性能的情况下做到了不用我们写.proto文件来实现序列化。使用它也非常简单,所以直接上代码。
- package gudao.red.protostuff;
- import java.util.Arrays;
- import java.util.List;
- public class Po {
- private Integer id;
- private String name;
- private String remark;
- private Integer age;
- private int[] array;
- private InnerPo innerPo;
- private List<String> more;
- public Po(){}
- public Po(Integer id,String name,String remark,Integer age,
- int[] array,InnerPo innerPo,List<String> more){
- this.id = id;
- this.name = name;
- this.remark = remark;
- this.age = age;
- this.array = array;
- this.innerPo = innerPo;
- this.more = more;
- }
- public Integer getId() {
- return id;
- }
- public void setId(Integer id) {
- this.id = id;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getRemark() {
- return remark;
- }
- public void setRemark(String remark) {
- this.remark = remark;
- }
- public Integer getAge() {
- return age;
- }
- public void setAge(Integer age) {
- this.age = age;
- }
- public int[] getArray() {
- return array;
- }
- public void setArray(int[] array) {
- this.array = array;
- }
- public InnerPo getInnerPo() {
- return innerPo;
- }
- public void setInnerPo(InnerPo innerPo) {
- this.innerPo = innerPo;
- }
- public List<String> getMore() {
- return more;
- }
- public void setMore(List<String> more) {
- this.more = more;
- }
- @Override
- public String toString() {
- StringBuffer sb = new StringBuffer();
- sb.append("id:"+id+"\n");
- sb.append("name:"+name+"\n");
- sb.append("remark:"+remark+"\n");
- sb.append("age:"+age+"\n");
- sb.append("array:"+Arrays.toString(array)+"\n");
- sb.append("innerPo:"+innerPo.toString()+"\n");
- sb.append("more:"+more);
- return sb.toString();
- }
- }
- class InnerPo{
- private Integer id;
- private String name;
- public InnerPo(){}
- public InnerPo(Integer id,String name){
- this.id = id;
- this.name = name;
- }
- public Integer getId() {
- return id;
- }
- public void setId(Integer id) {
- this.id = id;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- @Override
- public String toString() {
- return id+"-"+name;
- }
- }
- package gudao.red.protostuff;
- import java.util.ArrayList;
- import java.util.List;
- import com.dyuproject.protostuff.LinkedBuffer;
- import com.dyuproject.protostuff.ProtostuffIOUtil;
- import com.dyuproject.protostuff.runtime.RuntimeSchema;
- public class ProtostuffTest {
- static RuntimeSchema<Po> poSchema = RuntimeSchema.createFrom(Po.class);
- private static byte[] decode(Po po){
- return ProtostuffIOUtil.toByteArray(po, poSchema, LinkedBuffer.allocate(LinkedBuffer.DEFAULT_BUFFER_SIZE));
- }
- private static Po ecode(byte[] bytes){
- Po po = poSchema.newMessage();
- ProtostuffIOUtil.mergeFrom(bytes, po, poSchema);
- return po;
- }
- public static void main(String[] args) {
- InnerPo innerPo = new InnerPo(1, "InnerPo1");
- List<String> list = new ArrayList<String>();
- list.add("a");
- list.add("b");
- Po po = new Po(1, "Fong", "备注", 24, new int[]{1,2,3,4},innerPo,list);
- byte[] bytes = decode(po);
- System.out.println(bytes.length);
- Po newPo = ecode(bytes);
- System.out.println(newPo);
- }
- }
protobuf和protostuff的区别的更多相关文章
- java序列化/反序列化之xstream、protobuf、protostuff 的比较与使用例子
目录 背景 测试 环境 工具 说明 结果 结论 xstream简单教程 准备 代码 protobuf简单教程 快速入门 下载.exe编译器 编写.proto文件 利用编译器编译.proto文件生成ja ...
- protostuff简单应用
protobuf是谷歌推出的与语言无关.平台无关的通信协议,一个对象经过protobuf序列化后将变成二进制格式的数据,所以他可读性差,但换来的是占用空间小,速度快.居网友测试,它的序列化效率是xml ...
- 通讯协议序列化解读(二) protostuff详解教程
上一篇文章 通讯协议序列化解读(一):http://www.cnblogs.com/tohxyblog/p/8974641.html 前言:上一面文章我们介绍了java序列化,以及谷歌protobu ...
- Protostuff序列化分析
前言最近项目中需要将业务对象直接序列化,然后存数据库:考虑到序列化.反序列化的时间以及生产文件的大小觉得Protobuf是一个很好的选择,但是Protobuf有的问题就是需要有一个.proto的描述文 ...
- Protostuff具体解释
Protostuff具体解释 作者:chszs,未经博主同意不得转载. 经许可的转载需注明作者和博客主页:http://blog.csdn.net/chszs 一.Protostuff介绍 Proto ...
- Protobuf 小试牛刀
本文以PHP为例. 环境: CentOS 6.8 proto 3.8 PHP 7.1.12 PHP protobuf扩展 3.8.0 go1.12.5 linux/amd64 本文示例仓库地址: ht ...
- Protobuf的上手使用
这里不赘述Json和Protobuf的比较和区别,只谈谈简单的使用 1.在Client-Server交互的过程中,都是以二进制数据传输,所以Json和Protobuf在使用的过程中,都存在序列化和反序 ...
- protobuf 协议浅析
目录 Protobuf 协议浅析 1. Protobuf 介绍 1.1 Protobuf 基本概念 1.2 Protobuf 的优点 1.3 Protobuf, JSON, XML 的区别 2. Pr ...
- kafka消费者客户端
Kafka消费者 1.1 消费者与消费者组 消费者与消费者组之间的关系 每一个消费者都隶属于某一个消费者组,一个消费者组可以包含一个或多个消费者,每一条消息只会被消费者组中的某一个消费者所消费.不 ...
随机推荐
- poj 1182 (关系并查集) 食物链
题目传送门:http://poj.org/problem?id=1182 这是一道关系型并查集的题,对于每个动物来说,只有三种情况:同类,吃与被吃: 所以可以用0,1,2三个数字代表三种情况,在使用并 ...
- 如何搭建http服务仓库
1.拷贝仓库repo-A文件到服务器/media/D: 2.通过createrepo_c 生成仓库rpm信息数据 cd repo-A createrepo . 3.chmod -R 775 repo ...
- PS合成的5个要点:场景、对比、氛围、模糊、纹理
是否觉得做合成打开PS之后无处下手,做完之后总觉得缺少故事情节?这一次分享的5个要点,是个人觉得需要重视的,每一点都有一个案例来让作品变得多一份惊喜.(申明:文中素材均来自网络,这里仅作分享交流作用) ...
- [Robot Framework] 学习资料
https://www.cnblogs.com/pachongshangdexuebi/category/981644.html Robot Framework学习笔记(一)------环境搭建 Ro ...
- Spring 属性注入(一)JavaBean 内省机制在 BeanWrapper 中的应用
Spring 属性注入(一)JavaBean 内省机制在 BeanWrapper 中的应用 Spring 系列目录(https://www.cnblogs.com/binarylei/p/101174 ...
- eclipse项目两个红点
Description Resource Path Location Type Unbound classpath container: 'JRE Sy 选中项目右键build path 选择libr ...
- js中的find(),filter(),has()的用法和区别
filter():操作当前元素集,删除不匹配的元素,得到一个新的集合 <ul> <li class="a">list item 1</li> & ...
- jquery查找frameset框架内iframe的元素
老系统还幸存有过时的frameset框架,维护升级工作需要对其内部的iframe中的元素进行相关操作.使用jquery查找子iframe页面内的元素时,总找不到目标元素.后来发现少了contents ...
- vs2008 安装部署 启动项
具体操作办法如下: 鼠标右键安装项目->视图->注册表 依次创建键: HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion ...
- Android NDK定位.so文件crash代码位置
参考:http://blog.csdn.net/xyang81/article/details/42319789 问题: QRD8926_110202平台的Browser必现报错.(去年的项 ...