Kryo序列化與反序列化列子

這是一個Kryo將複雜Bean對象序列化與反序列化,通過字符串傳遞的列子。
具體的bean對象此處沒有附加代碼,可以自己寫一個bean的例子,裏面可包含任何對象,集合,基本類型。但要求有geter和setter方法。
本人在本地測試類以下,除了Map需要Setter方法外,其他屬性沒有setter方法也可序列化與反序列化。

這個例子中使用到了 Class BeanSerializer, 以下是這個類的介紹:
Serializes Java beans using bean accessor methods. Only bean properties with both a getter and setter are serialized. This class is not as fast as FieldSerializer but is much faster and more efficient than Java serialization. Bytecode generation is used to invoke the bean propert methods, if possible.

BeanSerializer does not write header data, only the object data is stored. If the type of a bean property is not final (note primitives are final) then an extra byte is written for that property.

Kryo官方文檔
Kryo3.03 jar包

TaskResult的類:
這裏寫圖片描述
TaskItemResult類:
這裏寫圖片描述

這裏寫圖片描述

package bhz.test;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.commons.codec.binary.Base64;

import bhz.test.task.ExecItemResultIPMI;
import bhz.test.task.IExecItemResult;
import bhz.test.task.ITaskResult;
import bhz.test.task.TaskItemResult;
import bhz.test.task.TaskResult;

import com.esotericsoftware.kryo.Kryo;
import com.esotericsoftware.kryo.io.Input;
import com.esotericsoftware.kryo.io.Output;
import com.esotericsoftware.kryo.serializers.BeanSerializer;
import com.esotericsoftware.kryo.serializers.JavaSerializer;
import com.esotericsoftware.kryo.serializers.MapSerializer;
import com.po.Animal;
import com.po.Dog;
import com.veraxsystems.vxipmi.coding.commands.ResponseData;
import com.veraxsystems.vxipmi.coding.commands.sdr.GetSdrResponseData;

public class KyroSerializableOne {

    public static void main(String[] args) throws IOException {
        /*
        long start = System.currentTimeMillis();
        Map<String, Integer> map = new HashMap<String, Integer>();
        map.put("zhang0", 2);
        map.put("zhang1", 3);
        Simple simple = new Simple("zhang" + 2, 2, map);
        //System.out.println(simple.getAge() + "  " + simple.getName() + "  "
        //      + simple.getMap().toString());
        ExecItemResultIPMI ipmiResult = new ExecItemResultIPMI();
        Dog dog = (Dog)simple.getDog();
        dog.getExecItemResult().add(ipmiResult);

        System.out.println("Kryo 序列化時間:" + (System.currentTimeMillis() - start)
                + " ms");
        start = System.currentTimeMillis();
        Dog dogSerial = (Dog)simple.getDog();
        IExecItemResult iExecItemResult = dogSerial.getExecItemResult().get(0);
        ExecItemResultIPMI ipmiResultSerial = (ExecItemResultIPMI)iExecItemResult;
        GetSdrResponseData getSdrResponseData = (GetSdrResponseData)ipmiResultSerial.getResponseData();

        //序列化
        String data= setSerializableObject(simple,Simple.class);

        //反序列化
        Simple simple1 = (Simple)getSerializableObject(Simple.class,data);



        System.out.println(simple1.getAge() + "  " + simple1.getName() + "  "
                + simple1.getMap().toString()+"   "+simple1.getDog().getName()+"  ipmiResultSerial :"+ipmiResultSerial.getResponseData()+
                "   getSdrResponseData:" +new String(getSdrResponseData.getSensorRecordData()));
        System.out.println("Kryo 反序列化時間:"
                + (System.currentTimeMillis() - start) + " ms");

        */




        /*GetSdrResponseData getSdrResponseData = new GetSdrResponseData();
        getSdrResponseData.setNextRecordId(111);
        getSdrResponseData.setSensorRecordData("getSdrResponseData".getBytes());
        ResponseData responseData = getSdrResponseData;
        */




        TaskResult taskResult = new TaskResult();
        Map<Integer, TaskItemResult> taskItemResult = taskResult.getTaskItemResult();

        TaskItemResult itemResult = new TaskItemResult();
        List<IExecItemResult> execItemResult = itemResult.getExecItemResult();

        ExecItemResultIPMI ipmiResult = new ExecItemResultIPMI();

        execItemResult.add(ipmiResult);
        taskItemResult.put(111, itemResult);


        System.out.println(taskResult);
        //ITaskResult result = taskResult;

        //System.out.println(result.getTaskItemResult());


        String setSerializableObject = setSerializableObject(taskResult,TaskResult.class);
        TaskResult resultObject = (TaskResult)getSerializableObject(TaskResult.class, setSerializableObject);

        Map<Integer, TaskItemResult> taskItemResult2 = resultObject.getTaskItemResult();
        System.out.println(taskItemResult2);

    }

    public  static <T> String setSerializableObject(Object object,Class<T> clazz ) throws FileNotFoundException {

        Kryo kryo = new Kryo();


        /*MapSerializer serializer = new MapSerializer();
            serializer.setKeyClass(Integer.class, new JavaSerializer());
            serializer.setKeysCanBeNull(false);
            serializer.setValueClass(TaskItemResult.class, new BeanSerializer(kryo, TaskItemResult.class));
            serializer.setValuesCanBeNull(true);
        kryo.register(HashMap.class, serializer);*/


        kryo.register(clazz, new BeanSerializer(kryo, clazz));

        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        Output output = new Output(baos);
        kryo.writeObject(output, object);
        output.flush();
        output.close();
        byte[] bys = baos.toByteArray();
        try {
            baos.flush();
            baos.close();
        } catch (IOException e) {
            e.printStackTrace();
        }

        String string = new String(new Base64().encode(bys));


        //getSerializableObject(string);
        System.out.println(string);

        return string;

    }

    public static <T> Object getSerializableObject(Class<T> clazz, String data) {
        Kryo kryo = new Kryo();



        /* MapSerializer serializer = new MapSerializer();
            serializer.setKeyClass(Integer.class, new JavaSerializer());
            serializer.setKeysCanBeNull(false);
            serializer.setValueClass(TaskItemResult.class, new BeanSerializer(kryo, TaskItemResult.class));
            serializer.setValuesCanBeNull(true);
            kryo.register(HashMap.class, serializer);*/



        kryo.register(clazz, new BeanSerializer(kryo, clazz));
        ByteArrayInputStream bais = new ByteArrayInputStream(new Base64().decode(data.getBytes()));
        Input input;
        Object simple = null;
        try {
            input = new Input(bais);
            input.close();
            simple = kryo.readObject(input, clazz);
            // if((simple=kryo.readObject(input, Simple.class)) != null){
            //System.out.println(simple.getAge() + "  " + simple.getName() + "  "
            //      + simple.getMap().toString()+"   "+simple.getDog().getName());
            // }

        } catch (Exception e) {
            e.printStackTrace();
        }
        return simple;
    }
}

更新:

以下是經過整理和運行驗證過的方法:
將需要序列化的Bean對象 ArrayList、 HashMap 放入方法中序列化,然後在反序列化,強轉成自己想要的類型即可。

以下是序列化反序列化的方法:

/**
     * 序列化bean對象的方法
     * @param object
     * @param clazz
     * @return
     * @throws FileNotFoundException
     */
    public  static <T> String setSerializableObject(Object object,Class<T> clazz ) throws FileNotFoundException {
        Kryo kryo = new Kryo();
        kryo.register(clazz, new BeanSerializer<T>(kryo, clazz));

        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        Output output = new Output(baos);
        kryo.writeObject(output, object);
        output.flush();
        output.close();
        byte[] bys = baos.toByteArray();
        try {
            baos.flush();
            baos.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        String string = new String(new Base64().encode(bys));

        System.out.println(string);

        return string;

    }


    /**
     * 反序列化bean對象的方法
     * @param clazz
     * @param data
     * @return
     */
    public static <T> Object getSerializableObject(Class<T> clazz, String data) {
        Kryo kryo = new Kryo();

        kryo.register(clazz, new BeanSerializer<T>(kryo, clazz)); //bean

        ByteArrayInputStream bais = new ByteArrayInputStream(new Base64().decode(data.getBytes()));
        Input input;
        Object simple = null;
        try {
            input = new Input(bais);
            input.close();
            simple = kryo.readObject(input, clazz);  //bean
            //simple = kryo.readObject(input, ArrayList.class, serializer);//list

        } catch (Exception e) {
            e.printStackTrace();
        }
        return simple;
    }


    /**
     * 序列化arrayList的方法
     * @param object
     * @param clazz
     * @return
     * @throws FileNotFoundException
     */
    public  static <T> String setSerializableList(Object object,Class<T> clazz ) throws FileNotFoundException {
        Kryo kryo = new Kryo();

        CollectionSerializer serializer = new CollectionSerializer();
        serializer.setElementClass(clazz, new BeanSerializer<T>(kryo, clazz));
        serializer.setElementsCanBeNull(false);

        kryo.register(clazz, new BeanSerializer<T>(kryo, clazz));
        kryo.register(ArrayList.class, serializer);


        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        Output output = new Output(baos);
        kryo.writeObject(output, object);
        output.flush();
        output.close();
        byte[] bys = baos.toByteArray();
        try {
            baos.flush();
            baos.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        String string = new String(new Base64().encode(bys));

        System.out.println(string);

        return string;

    }

    /**
     * 反序列化ArrayList的方法
     * @param clazz
     * @param data
     * @return
     */
    public static <T> Object getSerializableList(Class<T> clazz, String data) {
        Kryo kryo = new Kryo();
        //List
        CollectionSerializer serializer = new CollectionSerializer();
        serializer.setElementClass(clazz, new BeanSerializer<T>(kryo, clazz));
        serializer.setElementsCanBeNull(false);

        kryo.register(clazz, new BeanSerializer<T>(kryo, clazz));
        kryo.register(ArrayList.class, serializer);

        ByteArrayInputStream bais = new ByteArrayInputStream(new Base64().decode(data.getBytes()));
        Input input;
        Object simple = null;
        try {
            input = new Input(bais);
            input.close();
            simple = kryo.readObject(input, ArrayList.class, serializer);//list

        } catch (Exception e) {
            e.printStackTrace();
        }
        return simple;
    }

    /**
     * 序列化HashMap<String, clazz>的方法  key爲 String類型   value類型爲 clazz
     * @param object
     * @param clazz
     * @return
     * @throws FileNotFoundException
     */
    public  static <T> String setSerializableMap(Object object,Class<T> clazz ) throws FileNotFoundException {
        Kryo kryo = new Kryo();
        //設置序列化map的key和value類型
        MapSerializer serializer = new MapSerializer();
        serializer.setKeyClass(String.class, new JavaSerializer());
        serializer.setKeysCanBeNull(false);
        serializer.setValueClass(clazz, new BeanSerializer<T>(kryo, clazz));
        serializer.setValuesCanBeNull(true);
        //註冊要序列化的對象
        kryo.register(HashMap.class, serializer);
        kryo.register(clazz, new BeanSerializer<T>(kryo, clazz));

        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        Output output = new Output(baos);
        kryo.writeObject(output, object);
        output.flush();
        output.close();
        byte[] bys = baos.toByteArray();
        try {
            baos.flush();
            baos.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        String string = new String(new Base64().encode(bys));
        System.out.println(string);

        return string;

    }

    /**
     * HashMap<String,clazz>類型的map反序列化。
     * @param clazz
     * @param data
     * @return
     */
    public static <T> Object getSerializableMap(Class<T> clazz, String data) {
        Kryo kryo = new Kryo();


        //Map
        MapSerializer serializer = new MapSerializer();
        serializer.setKeyClass(String.class, new JavaSerializer());
        serializer.setKeysCanBeNull(false);
        serializer.setValueClass(clazz, new BeanSerializer<T>(kryo, clazz));
        serializer.setValuesCanBeNull(true);

        kryo.register(HashMap.class, serializer);
        kryo.register(clazz, new BeanSerializer<T>(kryo, clazz)); //bean

        ByteArrayInputStream bais = new ByteArrayInputStream(new Base64().decode(data.getBytes()));
        Input input;
        Object simple = null;
        try {
            input = new Input(bais);
            input.close();
            simple = kryo.readObject(input, HashMap.class, serializer);  //bean
        } catch (Exception e) {
            e.printStackTrace();
        }
        return simple;
    }
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章