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 消费者与消费者组 消费者与消费者组之间的关系 每一个消费者都隶属于某一个消费者组,一个消费者组可以包含一个或多个消费者,每一条消息只会被消费者组中的某一个消费者所消费.不 ...
随机推荐
- HTTP.ContentType
1. multipart/x-mixed-replace http://blog.dubbelboer.com/2012/01/08/x-mixed-replace.html
- 部分开源gis 方案的比较
3.1MapServer和GeoServer的总体对比 功能上:MapServer弱于GeoServer,QGIS要强于UDIG. 效率上:Mapserver对WMS(Web Map service) ...
- 201621123008 《Java程序设计》第12周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多流与文件相关内容. 2. 面向系统综合设计-图书馆管理系统或购物车 使用流与文件改造你的图书馆管理系统或购物车. 2.1 简述如何 ...
- hook 9大类
HOOK技术主要分为两大类,一是内核层HOOK,一是用户层HOOK. 用户层HOOK也就是在ring3环境下hook kenerl32.dll.User3.dll.Gui32.dll.Advapi.d ...
- Python的程序入口 __name__属性
python中每个模块都有一个 '__name__' 属性,当其值为 '__main__' 时,表名该模块自身在运行,否则是被引入的. 当一个模块被当做一个整体调用的时候,模块名.__name__ 的 ...
- String类为什么设计成不可变的
在Java中将String设计成不可变的是综合考虑到各种因素的结果,需要综合考虑内存.同步.数据结构以安全方面的考虑. String被设计成不可变的主要目的是为了安全和高效. 1)字符串常量池的需要 ...
- autohotkey快捷键
;已经基本修复了输入带shift的时候跟输入法中英文切换之间的冲突 SetStoreCapslockMode, off SetKeyDelay, ^CapsLock:: #UseHook ;用这个和下 ...
- openssl编译安装-各种蛋疼
arm平台 ubuntu Linux xxxxxx 3.0.62 #2 PREEMPT Tue Apr 2 20:14:12 CST 2013 armv7l armv7l armv7l GNU/Li ...
- MathExam
MathExam 一.预估与实际 PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时(分钟) Planning 计划 575 605 • Est ...
- js文件中获取${pageContext.request.contextPath}
一般从 JSP文件中,可以直接使用 ${pageContext.request.contextPath}非常方便的获得当前页面的路径,用来处理被 Apache2代理之后出现 URL变化的问题,比如增加 ...