Java泛型-3(实践篇-protostuff序列化与反序列化)

泛型学习目录:

Java泛型-1(泛型的定义)
Java泛型-2(通配符)
Java泛型-3(实践篇-protostuff序列化与反序列化)
Java泛型-4(类型擦除后如何获取泛型参数)

2. protostuff的准备工作

Java有一个序列化的技术,就是把Object转换为可保存,可传输的流数据。
Protostuff就是一个优秀的序列化框架。

1. 首先引入MAVEN依赖:

        <dependency>
            <groupId>io.protostuff</groupId>
            <artifactId>protostuff-runtime</artifactId>
        </dependency>

        <dependency>
            <groupId>io.protostuff</groupId>
            <artifactId>protostuff-core</artifactId>
        </dependency>

2. protostuff序列化和反序列化的问题:
使用protostuff的时候会遇到一些无法序列化/反序列化的对象,比如MapList等。

所以我们可以引入一个包装类来把数据包装下。

3. 代码分析

  • 我们引入静态内部类,因为该类只是为protostuff序列化进行服务的,故使用静态内部类。
  • 我们最好是在声明SerializeDeserializeWrapper<T>时,保存T的引用,以便后续逻辑中保存包装的类型
    //静态内部类
    public static class SerializeDeserializeWrapper<T> {
        //泛型的使用
        private T data;
        //建造者模式(返回实体类型)
        public static <T> SerializeDeserializeWrapper<T> builder(T data) {
            SerializeDeserializeWrapper<T> wrapper = new SerializeDeserializeWrapper<T>();
            wrapper.setData(data);
            return wrapper;
        }
        public void setData(T data) {
            this.data = data;
        }
        public T getData() {
            return data;
        }
    }

3.1 不使用泛型&&使用泛型

1.1 反序列化不使用泛型:

  public static Object deserialize(byte[] data, Class clazz) {
        try {
            //判断是否是不可序列化对象,若是不能序列化对象,将对象进行包装
            if (WRAPPER_SET.contains(clazz)) {
                //SerializeDeserializeWrapper<T> wrapper = SerializeDeserializeWrapper.builder(clazz.newInstance());
                SerializeDeserializeWrapper wrapper = new SerializeDeserializeWrapper();
                ProtostuffIOUtil.mergeFrom(data, wrapper, WRAPPER_SCHEMA);
                return  wrapper.getData();
            } else {
                Object message = clazz.newInstance();
                Schema<Object> schema = getSchema(clazz);
                ProtostuffIOUtil.mergeFrom(data, message, schema);
                return message;
            }
        } catch (Exception e) {
            logger.error("反序列化对象异常 [" + clazz.getName() + "]", e);
            throw new IllegalStateException(e.getMessage(), e);
        }
    }

1.2 测试方法

 Map data = (HashMap) MyProtostuffUtils.deserialize(serializer, map.getClass());

2.1 反序列化使用泛型:

public static <T> T deserialize(byte[] data, Class<T> clazz) {
        try {
            //判断是否是不可序列化对象,若是不能序列化对象,将对象进行包装
            if (WRAPPER_SET.contains(clazz)) {
                //SerializeDeserializeWrapper<T> wrapper = SerializeDeserializeWrapper.builder(clazz.newInstance());
                SerializeDeserializeWrapper<T> wrapper = new SerializeDeserializeWrapper<>();
                ProtostuffIOUtil.mergeFrom(data, wrapper, WRAPPER_SCHEMA);
                return wrapper.getData();
            } else {
                T message = clazz.newInstance();
                Schema<T> schema = getSchema(clazz);
                ProtostuffIOUtil.mergeFrom(data, message, schema);
                return message;
            }
        } catch (Exception e) {
            logger.error("反序列化对象异常 [" + clazz.getName() + "]", e);
            throw new IllegalStateException(e.getMessage(), e);
        }
    }

2.2 测试方法:

String data = MyProtostuffUtils.deserialize(serializer, str.getClass());

3. 对比

两个都可执行成功,但是使用泛型的话,

  1. 编译器就可以进行类型检查,注意在运行期会进行泛型擦除会找到用来替换类型参数的具体类。例如SerializeDeserializeWrapper<String>对象,运行期泛型擦除class文件中的T都会转化为String类型;SerializeDeserializeWrapper对象,运行期class文件中的T将会被转化为Object类型。
  2. 因为返回值是泛型T,那么无需进行类型转换。
  3. Class<T>传入的class对象后,我们就可以拿到T的也就是class对象的类型。这个传入的T,可以在代码中任意地方使用。

3.2 源码分析

1. 源码分析:

  • 将需要保存的信息保存到Set<Class<?>>里面,在类初始化的时候,使用static进行加载。将需要包装的类型,保存到集合里面。
  • 泛型通配符中<?>代表的是未知类型。可以接受Class<XXX>所有的类型。注意:Class<Object>不能接收Class<String>等类型,因为他们不是父子类的关系。
  • 需要一个Cache保存Class<?>Schema<?>的关系,故可以使用ConcurrentHashMap,保存映射关系。
  • 因为使用的包装类型SerializeDeserializeWrapper,那么他们的class类型和Schema都是固定的。
  • 用户的参数是泛型T-Type(而不是Object)对象,那么便可以获取到对应的class对象以及schema对象。

2. 源码如下

package com.protoType;

import io.protostuff.LinkedBuffer;
import io.protostuff.ProtostuffIOUtil;
import io.protostuff.Schema;
import io.protostuff.runtime.RuntimeSchema;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;

/***
 * CACHE_SCHEMA  缓冲区。K-V 分别是class(类结构信息)对象和Schema(模板)对象
 *
 */
public class MyProtostuffUtils {
    private static Logger logger = LoggerFactory.getLogger(MyProtostuffUtils.class);
    //将数据封装
    private static final Set<Class<?>> WRAPPER_SET = new HashSet<>();

    //包装类的Class对象
    private static final Class<SerializeDeserializeWrapper> WRAPPER_CLASS = SerializeDeserializeWrapper.class;

    //包装类的Schema对象
    private static final Schema<SerializeDeserializeWrapper> WRAPPER_SCHEMA = RuntimeSchema.createFrom(WRAPPER_CLASS);

    //安全缓存区,class对象和Schema对象
    private static final Map<Class<?>, Schema<?>> CACHE_SCHEMA = new ConcurrentHashMap<>();

    static {
        WRAPPER_SET.add(List.class);
        WRAPPER_SET.add(ArrayList.class);
        WRAPPER_SET.add(CopyOnWriteArrayList.class);
        WRAPPER_SET.add(LinkedList.class);
        WRAPPER_SET.add(Stack.class);
        WRAPPER_SET.add(Vector.class);
        WRAPPER_SET.add(Map.class);
        WRAPPER_SET.add(HashMap.class);
        WRAPPER_SET.add(TreeMap.class);
        WRAPPER_SET.add(LinkedHashMap.class);
        WRAPPER_SET.add(Hashtable.class);
        WRAPPER_SET.add(SortedMap.class);
//        WRAPPER_SET.add(Object.class);
    }

    //注册需要使用包装类进行序列化的Class对象
    public static void registerWrapperClass(Class clazz) {
        WRAPPER_SET.add(clazz);
    }

    //获取序列化对象类型的schema
    private static <T> Schema<T> getSchema(Class<T> clazz) {
        Schema<T> schema = (Schema<T>) CACHE_SCHEMA.get(clazz);
        if (schema == null) {
            schema = RuntimeSchema.createFrom(clazz);
            CACHE_SCHEMA.put(clazz, schema);
        }
        return schema;
    }

    //序列化对象
    public static <T> byte[] serializer(T obj) {
        //获取序列化对象
        Class<T> clazz = (Class<T>) obj.getClass();
        //设置缓数组缓冲区
        LinkedBuffer buffer = LinkedBuffer.allocate(LinkedBuffer.DEFAULT_BUFFER_SIZE);
        byte[] bytes = null;
        try {
            Object serializerObj = obj;       //获取序列化对象
            Schema schema = WRAPPER_SCHEMA;   //获取Schema对象
            //包装class对象
            if (WRAPPER_SET.contains(clazz)) {
                //外部类是否可以使用静态内部类的成员?【外部类使用内部类的成员,需要新建内部类实例。】
                serializerObj = SerializeDeserializeWrapper.builder(obj);//将class对象进行包装
            } else {
                //将class对象和schema对象保存到hashMap中
                schema = getSchema(clazz);  //获取Schema对象
            }
            //将对象转换为字节流
            bytes = ProtostuffIOUtil.toByteArray(serializerObj, schema, buffer);
        } catch (Exception e) {
            logger.info("序列化{}失败", obj, e);
            throw new IllegalStateException(e.getMessage());
        } finally {
            //回收buffer
            buffer.clear();
        }
        return bytes;
    }


    public static <T> T deserialize(byte[] data, Class<T> clazz) {
        try {
            //判断是否是不可序列化对象,若是不能序列化对象,将对象进行包装
            if (WRAPPER_SET.contains(clazz)) {
                //SerializeDeserializeWrapper<T> wrapper = SerializeDeserializeWrapper.builder(clazz.newInstance());
                SerializeDeserializeWrapper<T> wrapper = new SerializeDeserializeWrapper<>();
                ProtostuffIOUtil.mergeFrom(data, wrapper, WRAPPER_SCHEMA);
                return wrapper.getData();
            } else {
                T message = clazz.newInstance();
                Schema<T> schema = getSchema(clazz);
                ProtostuffIOUtil.mergeFrom(data, message, schema);
                return message;
            }
        } catch (Exception e) {
            logger.error("反序列化对象异常 [" + clazz.getName() + "]", e);
            throw new IllegalStateException(e.getMessage(), e);
        }
    }


    //静态内部类
    public static class SerializeDeserializeWrapper<T> {
        //泛型的使用
        private T data;

        //建造者模式(返回实体类型)
        public static <T> SerializeDeserializeWrapper<T> builder(T data) {
            SerializeDeserializeWrapper<T> wrapper = new SerializeDeserializeWrapper<T>();
            wrapper.setData(data);
            return wrapper;
        }

        public void setData(T data) {
            this.data = data;
        }

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

推荐阅读更多精彩内容