以前使用 protobuf或protostuff的时候觉得很麻烦,每个类都要单独定制,于是封装了一个类。

同事测试过,性能和压缩率都很好,尤其是相比json的序列化。

需注意:只支持Pojo类(即需要有get/set方法)、对一个新的class第一次调用初始化会有一两百毫秒的register时间,之后就很快了。

import io.protostuff.LinkedBuffer;
import io.protostuff.ProtostuffIOUtil;
import io.protostuff.Schema;
import io.protostuff.runtime.RuntimeSchema; import java.io.Serializable;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; /**
* Protostuff serializer tool, for POJO serialization.
* Protostuff is much more efficient than json, even faster than Protobuf and Avro, but the serialized string is human-unreadable.
* Not support Array or Generic-type, please wrap these special objects via a POJO with empty constructors.
*
* @author lhfcws
* @since 2016-03-16
*/
public class ProtostuffSerializer implements Serializable { static Map<Class, Schema> schemaCache = new ConcurrentHashMap<>(); /**
* common protostuff serialize, object need a empty constructor
* Be careful to convert result byte[] to String, use new String(bytes, StandardCharsets.UTF_16LE).
*
* @param obj
* @param <T>
* @return
*/
public static <T> byte[] serializeObject(T obj) {
Class<T> klass = (Class<T>) obj.getClass();
LinkedBuffer buffer = LinkedBuffer.allocate(4096);
try {
if (schemaCache.containsKey(klass)) {
return ProtostuffIOUtil.toByteArray(obj, schemaCache.get(klass), buffer);
} else {
schemaCache.put(klass, RuntimeSchema.getSchema(klass));
return ProtostuffIOUtil.toByteArray(obj, schemaCache.get(klass), buffer);
}
} finally {
buffer.clear();
}
} /**
* common protostuff unserialize
*
* @param bs
* @param klass
* @param <T>
* @return
*/
public static <T> T deserialize(byte[] bs, Class<T> klass) {
if (schemaCache.containsKey(klass)) {
Schema<T> schema = schemaCache.get(klass);
T msg = schema.newMessage();
ProtostuffIOUtil.mergeFrom(bs, msg, schema);
return msg;
} else {
Schema<T> schema = RuntimeSchema.getSchema(klass);
T msg = schema.newMessage();
schemaCache.put(klass, schema);
ProtostuffIOUtil.mergeFrom(bs, msg, schema);
return msg;
}
}
}

使用demo:

// 如果是Pojo类直接调用就行了,非Pojo类参考如下:(假设已有一个StrParams model类)

    public static class StrParamsPojo {
private StrParams p; public StrParamsPojo() {
} public StrParamsPojo(StrParams p) {
this.p = p;
} public StrParams getP() {
return p;
} public void setP(StrParams p) {
this.p = p;
}
} public void serialize() throws IOException {
StrParams p = new StrParams();
StrParamsPojo pojo = new StrParamsPojo(p);
byte[] bs = ProtostuffSerializer.serializeObject(pojo);
} public void deserialize(byte[] bs) throws IOException {
StrParamsPojo pojo = ProtostuffSerializer.deserialize(bs, StrParamsPojo.class);
StrParams p = pojo.getP();
}

附送一个FastJsonSerializer:

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.serializer.SerializerFeature; import java.lang.reflect.Type; /**
* FastJson is faster than Gson.
* But DO remember your objects has get/set for the fields you want to serialze.
* @author lhfcws
*/
public class FastJsonSerializer { /**
* 把给定的对象序列化成json字符串
* @param obj 给定的对象
* @return 对象序列化后的json字符串
*/
public static <T> String serialize(T obj) {
return JSON.toJSONString(obj,
SerializerFeature.IgnoreNonFieldGetter,
SerializerFeature.SkipTransientField,
SerializerFeature.DisableCircularReferenceDetect,
SerializerFeature.BrowserCompatible
);
} public static <T> String serializePretty(T obj) {
return JSON.toJSONString(obj,
SerializerFeature.IgnoreNonFieldGetter,
SerializerFeature.SkipTransientField,
SerializerFeature.DisableCircularReferenceDetect,
SerializerFeature.BrowserCompatible,
SerializerFeature.PrettyFormat
);
} /**
* 根据类名把json字符串反序列化成实体类对象
* @param json 待反序列化的json字符串
* @param klass 反序列化的实体类
* @return 反序列化后的对象
*/
public static <T> T deserialize(String json, Class<T> klass) {
return JSON.parseObject(json, klass);
} /**
* 把Json字符串反序列化成实现了Type接口的实体类对象
* @param json 待反序列化的json字符串
* @param type 泛型类型
* @return 反序列化后的对象
*/
public static <T> T deserialize(String json, Type type) {
return JSON.parseObject(json, type);
}
}

通用的ProtostuffSerializer for Java的更多相关文章

  1. 通用且常用的Java正则匹配工具,用以检查邮箱名、电话号码、用户密码、邮政编码等合法性

    一个通用且常用的Java正则匹配工具,用以检查邮箱名.电话号码.用户密码.邮政编码等合法性. import java.util.regex.Matcher; import java.util.rege ...

  2. AES加密解密通用版Object-C / C# / JAVA

    1.无向量 128位 /// <summary> /// AES加密(无向量) /// </summary> /// <param name="plainByt ...

  3. 编写通用shell脚本启动java项目,适用于多数服务,只需修改服务名即可

    文件名:service-user.sh 文件内容: ##shell脚本的头文件必须有#!/bin/sh ##再次配置java环境变量以防报其他错误## java env#jdk安装目录export J ...

  4. 通用RSA加密 - PHP+Java+Javascript加密解密

    php端生成 公钥私钥 1.openssl genrsa -out rsa_private_key.pem 1024    私钥 2.openssl rsa -in rsa_private_key.p ...

  5. Java简单实现AOP,Java通用异常拦截,Java与Lamada

    直接看代码不废话.不懂Lamada直接百度... package test; /** * QQ:1448376744 * @author 花间岛 * */ //控制器 public class Con ...

  6. Java通用oracle和mysql数据库连接

    Java中oracle数据库连接写一个通用类UBUtil(){} import java.io.InputStream; import java.sql.*; import java.util.Pro ...

  7. Effective Java:对于全部对象都通用的方法

    前言: 读这本书第1条规则的时候就感觉到这是一本非常好的书.可以把我们的Java功底提升一个档次,我还是比較推荐的.这里我主要就关于覆盖equals.hashCode和toString方法来做一个笔记 ...

  8. 浅析Java 泛型

    泛型是JavaSE5引入的一个新概念,但是这个概念在编程语言中却是很普遍的一个概念.下面,根据以下内容,我们总结下在Java中使用泛型. 泛型使用的意义 什么是泛型 泛型类 泛型方法 泛型接口 泛型擦 ...

  9. Java JNI 编程进阶 实例+c++数据类型与jni数据类型转换

    原文:http://www.iteye.com/topic/295776 JNI一直以来都很少去关注,但却是我心中的一个结,最近这几天刚好手头有点时间,因此抽空看了一下这方面的东西,整理了一份文档,J ...

随机推荐

  1. selenium相关:通过location 和 size 获取元素所在像素位置和尺寸,截取图片ROI

    1.实验 #https://captcha.luosimao.com/demo/ chrome default: location 不滚动,直接返回相对整个html的坐标 {'x': 15.0, 'y ...

  2. Django项目和Django初体验和创建、目录结构认识

    .MVC的设计方式(跟Flask一样,都是MVC的设计模式) .开发效率高 .功能强大(丰富的第三方组件) .安全性高(帮助开发者规避安全漏洞) 目前市面上使用:Django>Flask #使用 ...

  3. easyui 信息提示

    /*消息提示begin*/jQuery.Info = function (msg) { $.messager.alert("温馨提示", msg, "info" ...

  4. P1182 数列分段`Section II` P1316 丢瓶盖 二分答案

    题目描述 对于给定的一个长度为N的正整数数列A-iA−i,现要将其分成M(M≤N)M(M≤N)段,并要求每段连续,且每段和的最大值最小. 关于最大值最小: 例如一数列4 2 4 5 142451要分成 ...

  5. ES集群

    1. ElasticSerach集群安装  修改配置文件elasticserach.yml [elk@localhost config]$ vi elasticsearch.yml # ------- ...

  6. OpenJ_Bailian 4017 爬楼梯

    时间限制: 1000 ms  空间限制: 262144 KB 题目描述 树老师爬楼梯,他可以每次走1级或者2级,输入楼梯的级数,求不同的走法数.例如:楼梯一共有3级,他可以每次都走一级,或者第一次走一 ...

  7. 我今天遇到的条件语句Integer类型的

    两个Integer类型的值进行比较时,应该用equals进行判断,用"=="判断是错误的,后来想了一下就明白了,Integer毕竟是对象, 而不是int基本数据类型,可以直接比较, ...

  8. HDU 1175 连连看 (DFS+剪枝)

    <题目链接> 题目大意:在一个棋盘上给定一个起点和终点,判断这两点是否能通过连线连起来,规定这个连线不能穿过其它的棋子,并且连线转弯不能超过2次. 解题分析:就是DFS从起点开始搜索,只不 ...

  9. POJ 2001 Shortest Prefixes 【Trie树】

    <题目链接> 题目大意: 找出能唯一标示一个字符串的最短前缀,如果找不出,就输出该字符串. 解题分析: Trie树的简单应用,对于每个单词的插入,都在相应字符对应的节点 num 值+1 , ...

  10. FTP传输协议的应用详解

    FTP的目标:1)促进程序.数据文件按的共享;2)鼓励使用远程计算机;3)使用户不必面对不同主机上不同文件系统的差异;4)对数据进行高效可靠的传输FTP的作用:就是让用户连接上一个远程计算机,察看远程 ...