SpringBoot--實戰(zhàn)開發(fā)--Protocol Buffer(五十四)

一笑窜、Protostuff簡介

??Protocol Buffer是谷歌出品的一種數(shù)據(jù)交換格式致燥,獨(dú)立于語言和平臺登疗,類似于json。Google提供了多種語言的實現(xiàn):java嫌蚤、c++辐益、go和python。對象序列化城Protocol Buffer之后可讀性差脱吱,但是相比xml智政,json,它占用小箱蝠,速度快续捂。適合做數(shù)據(jù)存儲或 RPC 數(shù)據(jù)交換格式。
Java序列化庫 - Protostuff
相對我們常用的json來說宦搬,Protocol Buffer門檻更高牙瓢,因為需要編寫.proto文件,再把它編譯成目標(biāo)語言间校,這樣使用起來就很麻煩矾克。但是現(xiàn)在有了protostuff之后,就不需要依賴.proto文件了憔足,他可以直接對POJO進(jìn)行序列化和反序列化胁附,使用起來非常簡單酒繁。

二、Maven依賴

      <dependency>
           <groupId>io.protostuff</groupId>
           <artifactId>protostuff-core</artifactId>
           <version>${protostuff.version}</version>
       </dependency>

       <dependency>
           <groupId>io.protostuff</groupId>
           <artifactId>protostuff-runtime</artifactId>
           <version>${protostuff.version}</version>
       </dependency>

三控妻、

先編寫兩個POJO州袒,再把它們嵌套起來,這里使用了lombok的@Data注解和@Builder注解弓候,@Data可以自動生成getter setter稳析,@Builder注解可以讓我們通過更加優(yōu)雅的構(gòu)建者模式來創(chuàng)建對象。

@Data
@Builder
public class User {
    private String id;
    private String name;
    private Integer age;
    private String desc;
}

@Data
@Builder
public class Group {
    private String id;

    private String name;

    private User user;
}

public class ProtostuffUtils {
    /**
     * 避免每次序列化都重新申請Buffer空間
     */
    private static LinkedBuffer buffer = LinkedBuffer.allocate(LinkedBuffer.DEFAULT_BUFFER_SIZE);
    /**
     * 緩存Schema
     */
    private static Map<Class<?>, Schema<?>> schemaCache = new ConcurrentHashMap<>();

    /**
     * 序列化方法弓叛,把指定對象序列化成字節(jié)數(shù)組
     *
     * @param obj
     * @param <T>
     * @return
     */
    @SuppressWarnings("unchecked")
    public static <T> byte[] serialize(T obj) {
        Class<T> clazz = (Class<T>) obj.getClass();
        Schema<T> schema = getSchema(clazz);
        byte[] data;
        try {
            data = ProtostuffIOUtil.toByteArray(obj, schema, buffer);
        } finally {
            buffer.clear();
        }

        return data;
    }

    /**
     * 反序列化方法彰居,將字節(jié)數(shù)組反序列化成指定Class類型
     *
     * @param data
     * @param clazz
     * @param <T>
     * @return
     */
    public static <T> T deserialize(byte[] data, Class<T> clazz) {
        Schema<T> schema = getSchema(clazz);
        T obj = schema.newMessage();
        ProtostuffIOUtil.mergeFrom(data, obj, schema);
        return obj;
    }

    @SuppressWarnings("unchecked")
    private static <T> Schema<T> getSchema(Class<T> clazz) {
        Schema<T> schema = (Schema<T>) schemaCache.get(clazz);
        if (Objects.isNull(schema)) {
            //這個schema通過RuntimeSchema進(jìn)行懶創(chuàng)建并緩存
            //所以可以一直調(diào)用RuntimeSchema.getSchema(),這個方法是線程安全的
            schema = RuntimeSchema.getSchema(clazz);
            if (Objects.nonNull(schema)) {
                schemaCache.put(clazz, schema);
            }
        }

        return schema;
    }
}

驗證序列化功能

@SpringBootApplication
public class Application implements CommandLineRunner {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }

    @Override
    public void run(String... strings) throws Exception {
        //創(chuàng)建一個user對象
        User user = User.builder().id("1").age(20).name("張三").desc("programmer").build();
        //創(chuàng)建一個Group對象
        Group group = Group.builder().id("1").name("分組1").user(user).build();
        //使用ProtostuffUtils序列化
        byte[] data = ProtostuffUtils.serialize(group);
        System.out.println("序列化后:" + Arrays.toString(data));
        Group result = ProtostuffUtils.deserialize(data, Group.class);
        System.out.println("反序列化后:" + result.toString());
    }
}

可以看到控制臺打印出如下數(shù)據(jù),說明序列化和反序列化成功

序列化后:[10, 1, 49, 18, 7, -27, -120, -122, -25, -69, -124, 49, 27, 10, 1, 49, 18, 6, -27, -68, -96, -28, -72, -119, 24, 20, 34, 10, 112, 114, 111, 103, 114, 97, 109, 109, 101, 114, 28]
反序列化后:Group(id=1, name=分組1, user=User(id=1, name=張三, age=20, desc=programmer))

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末撰筷,一起剝皮案震驚了整個濱河市陈惰,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌毕籽,老刑警劉巖抬闯,帶你破解...
    沈念sama閱讀 218,284評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異关筒,居然都是意外死亡溶握,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評論 3 395
  • 文/潘曉璐 我一進(jìn)店門蒸播,熙熙樓的掌柜王于貴愁眉苦臉地迎上來睡榆,“玉大人,你說我怎么就攤上這事袍榆≌陀欤” “怎么了?”我有些...
    開封第一講書人閱讀 164,614評論 0 354
  • 文/不壞的土叔 我叫張陵包雀,是天一觀的道長宿崭。 經(jīng)常有香客問我,道長才写,這世上最難降的妖魔是什么葡兑? 我笑而不...
    開封第一講書人閱讀 58,671評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮赞草,結(jié)果婚禮上讹堤,老公的妹妹穿的比我還像新娘。我一直安慰自己房资,他們只是感情好蜕劝,可當(dāng)我...
    茶點故事閱讀 67,699評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般岖沛。 火紅的嫁衣襯著肌膚如雪暑始。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,562評論 1 305
  • 那天婴削,我揣著相機(jī)與錄音廊镜,去河邊找鬼。 笑死唉俗,一個胖子當(dāng)著我的面吹牛嗤朴,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播虫溜,決...
    沈念sama閱讀 40,309評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼雹姊,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了衡楞?” 一聲冷哼從身側(cè)響起吱雏,我...
    開封第一講書人閱讀 39,223評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎瘾境,沒想到半個月后歧杏,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,668評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡迷守,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,859評論 3 336
  • 正文 我和宋清朗相戀三年犬绒,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片兑凿。...
    茶點故事閱讀 39,981評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡凯力,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出急膀,到底是詐尸還是另有隱情沮协,我是刑警寧澤,帶...
    沈念sama閱讀 35,705評論 5 347
  • 正文 年R本政府宣布卓嫂,位于F島的核電站,受9級特大地震影響聘殖,放射性物質(zhì)發(fā)生泄漏晨雳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,310評論 3 330
  • 文/蒙蒙 一奸腺、第九天 我趴在偏房一處隱蔽的房頂上張望餐禁。 院中可真熱鬧,春花似錦突照、人聲如沸帮非。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,904評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽末盔。三九已至筑舅,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間陨舱,已是汗流浹背翠拣。 一陣腳步聲響...
    開封第一講書人閱讀 33,023評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留游盲,地道東北人误墓。 一個月前我還...
    沈念sama閱讀 48,146評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像益缎,于是被迫代替她去往敵國和親谜慌。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,933評論 2 355

推薦閱讀更多精彩內(nèi)容