Jackson多態(tài)序列化圖文詳解
場(chǎng)景
做一個(gè)消息中心,專門負(fù)責(zé)發(fā)送消息。消息分為幾種渠道,包括手機(jī)通知(Push)、短信(SMS)、郵件(Email),Websocket等渠道。
我定義了一個(gè)基類MessageRequest
用來接收請(qǐng)求參數(shù),代碼如下:
public class MessageRequest implements Serializable { protected MessageChannel channel; private MessageRequest(){} protected MessageRequest(MessageChannel channel){ this.channel = channel; } public MessageChannel getChannel() { return this.channel; } }
在MessageRequest
中有個(gè)屬性channel
是枚舉MessageChannel
,該枚舉列舉所有渠道,代碼如下:
public enum MessageChanne { PUSH, EMAIL, WEBSOCKET, SMS, ; MessageChannel() {} }
MessageRequest
有各種渠道的子類實(shí)現(xiàn),以Push為例:
public class PushMessageReuqest extends MessageRequest { public PushMessageRequest() { super(MessageChannel.PUSH); } private String title; // 省略其他字段以及getter、setter方法 ... }
我在接口入?yún)⑹褂?code>MessageRequest接收:
public class MessageController { @PostMapping("/sendMessage") public R<Object> sendMessage(MessageRequest request) { System.out.println(request); } }
使用postman發(fā)送push請(qǐng)求之后發(fā)現(xiàn)后端收到的類型還是基類,并且title字段丟失。
這與我預(yù)想的不符,因?yàn)榭蛻舳酥狼溃瑯?gòu)建對(duì)應(yīng)的渠道消息體給我就好了??!為什么類型被擦除了呢?我的想法就是發(fā)送push請(qǐng)求啊。。。。。后來才知道序列化之后在反序列化的時(shí)候不知道給你反序列化成什么類型,序列化工具也沒有聰明到能根據(jù)你的channel屬性就知道是什么類型,但是我又想這樣做。那么怎么辦呢????
Jackson多態(tài)類型序列化/反序列化
經(jīng)過查詢資料以及咨詢了一下領(lǐng)導(dǎo),發(fā)現(xiàn)了@JsonTypeInfo
和@JsonSubTypes
兩個(gè)注解。
@JsonTypeInfo
作用于類/接口,被用來開啟多態(tài)類型處理,它有一些屬性:
- use(必選):定義使用哪一種類型標(biāo)識(shí)碼,有以下幾個(gè)可選項(xiàng)。
NONE
:不使用識(shí)別碼CLASS
:使用完全限定類名做識(shí)別碼MINIMAL_CLASS
:使用類名(忽略包名)做識(shí)別碼,和基類在同一個(gè)包可用NAME
:指定名稱CUSTOM
:自定義識(shí)別碼,由@JsonTypeIdResolver
對(duì)應(yīng)
- include(可選):指定識(shí)別碼如何被包含進(jìn)去,有以下幾個(gè)可選項(xiàng)。
PROPERTY
:作為兄弟屬性加入,默認(rèn)值WRAPPER_OBJECT
:作為一個(gè)包裝的對(duì)象WRAPPER_ARRAY
:作為包裝的數(shù)組EXTERNAL_PROPERTY
:作為擴(kuò)展屬性EXISTING_PROPERTY
:作為已存在的屬性(符合我的場(chǎng)景,用channel)
- property(可選):指定識(shí)別碼的屬性名稱。該屬性只有當(dāng)
use
為CLASS(不指定默認(rèn)為@class
)、MINIMAL_CLASS(不指定默認(rèn)為@c
)、NAME(不指定默認(rèn)為@type
,include
為PROPERTY、EXISTING_PROPERTY、EXTERNAL_PROPERTY時(shí)才有效。 - defaultImpl(可選):如果類型識(shí)別碼不存在或者無(wú)效,可以使用該屬性來指定反序列化時(shí)使用的默認(rèn)類型。
- visible(可選,默認(rèn)false):屬性定義了類型標(biāo)識(shí)符是否會(huì)成為反序列化器的一部分,默認(rèn)為false,也就是說Jackson會(huì)從json內(nèi)容中刪除類型標(biāo)識(shí)再傳遞給JsonDeserializer。
@JsonSubTypes
作用于類/接口,用來列出給定類/接口的子類。一般配合@JsonTypeInfo
使用
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.PROPERTY, property = "channel") @JsonSubTypes({ @JsonSubTypes.Type(value = PushMessageRequest.class, name = "PUSH"), @JsonSubTypes.Type(value = EmailMessageRequest.class, name = "EMAIL") })
JsonSubTypes
的值是一個(gè)@JsonSubTypes.Type[]
數(shù)組,參數(shù)value
表示類型,參數(shù)name
表示@JsonTypeInfo
注解中property
屬性的值,對(duì)比以上代碼即:channel = "PUSH"或channel = "EMAIL"。name
為可選值,不指定時(shí)需在子類提供JsonTypeName
注解并指定value
屬性。
實(shí)戰(zhàn)
改造上面提供的MessageReuqest
// include默認(rèn)為PROPERTY,這里可以不加 @JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.PROPERTY, property = "channel") @JsonSubTypes({ @JsonSubTypes.Type(value = PushMessageRequest.class, name = "PUSH"), @JsonSubTypes.Type(value = EmailMessageRequest.class, name = "EMAIL") }) public class MessageRequest implements Serializable { protected MessageChannel channel; private MessageRequest(){} protected MessageRequest(MessageChannel channel){ this.channel = channel; } public MessageChannel getChannel() { return this.channel; } }
此時(shí)通過postman請(qǐng)求發(fā)現(xiàn)入?yún)㈩愋陀辛俗兓?/p>
include屬性使用默認(rèn)的PROPERTY時(shí)發(fā)現(xiàn)序列化之后的json會(huì)多出來一個(gè)屬性,屬性名對(duì)應(yīng)的就是@JsonTypeInfo
的property
的值。雖然不影響使用,但是我看著很不舒服?;谖疫@種情況可以使用include=EXISTING_PROPERTY
。
總結(jié)
到此這篇關(guān)于Jackson多態(tài)序列化的文章就介紹到這了,更多相關(guān)Jackson多態(tài)序列化內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用正則表達(dá)式判斷獨(dú)立字符的存在(代碼示例)
通過使用正則表達(dá)式,我們可以更加靈活地判斷字符串中是否包含特定的字符,并且可以控制匹配的條件,如獨(dú)立的字符,這為我們處理字符串提供了更多的選擇和功能,這篇文章主要介紹了Java使用正則表達(dá)式判斷獨(dú)立字符的存在,需要的朋友可以參考下2023-10-10Java?數(shù)據(jù)結(jié)構(gòu)深入理解ArrayList與順序表
ArrayList?類是一個(gè)可以動(dòng)態(tài)修改的數(shù)組,與普通數(shù)組的區(qū)別就是它是沒有固定大小的限制,我們可以添加或刪除元素。ArrayList?繼承了?AbstractList?,并實(shí)現(xiàn)了?List?接口,順序表是將元素順序地存放在一塊連續(xù)的存儲(chǔ)區(qū)里,元素間的順序關(guān)系由它們的存儲(chǔ)順序自然表示2022-04-04Mybatis傳入List實(shí)現(xiàn)批量更新的示例代碼
這篇文章主要介紹了Mybatis傳入List實(shí)現(xiàn)批量更新的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10SpringBoot中的@ControllerAdvice注解原理詳解
這篇文章主要介紹了SpringBoot中的@ControllerAdvice注解原理詳解,在SpringBoot應(yīng)用程序啟動(dòng)過程中,Spring會(huì)掃描所有的類,尋找?guī)в蠤ControllerAdvice注解的類這些方法會(huì)被添加到一個(gè)映射表中,以便后續(xù)處理異常時(shí)能找到對(duì)應(yīng)的處理方法,需要的朋友可以參考下2024-01-01springBoot啟動(dòng)時(shí)讓方法自動(dòng)執(zhí)行的幾種實(shí)現(xiàn)方式
這篇文章主要介紹了springBoot啟動(dòng)時(shí)讓方法自動(dòng)執(zhí)行的幾種實(shí)現(xiàn)方式,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03Java基于rest assured實(shí)現(xiàn)接口測(cè)試過程解析
這篇文章主要介紹了Java基于rest assured實(shí)現(xiàn)接口測(cè)試過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03spring+mybatis實(shí)現(xiàn)圖書管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了spring+mybatis實(shí)現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-06-06