Jackson多態(tài)序列化圖文詳解
場景
做一個(gè)消息中心,專門負(fù)責(zé)發(fā)送消息。消息分為幾種渠道,包括手機(jī)通知(Push)、短信(SMS)、郵件(Email),Websocket等渠道。
我定義了一個(gè)基類MessageRequest用來接收請求參數(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請求之后發(fā)現(xiàn)后端收到的類型還是基類,并且title字段丟失。


這與我預(yù)想的不符,因?yàn)榭蛻舳酥狼?,?gòu)建對應(yīng)的渠道消息體給我就好了?。槭裁搭愋捅徊脸四??我的想法就是發(fā)送push請求啊。。。。。后來才知道序列化之后在反序列化的時(shí)候不知道給你反序列化成什么類型,序列化工具也沒有聰明到能根據(jù)你的channel屬性就知道是什么類型,但是我又想這樣做。那么怎么辦呢????
Jackson多態(tài)類型序列化/反序列化
經(jīng)過查詢資料以及咨詢了一下領(lǐng)導(dǎo),發(fā)現(xiàn)了@JsonTypeInfo和@JsonSubTypes兩個(gè)注解。
@JsonTypeInfo作用于類/接口,被用來開啟多態(tài)類型處理,它有一些屬性:
- use(必選):定義使用哪一種類型標(biāo)識碼,有以下幾個(gè)可選項(xiàng)。
NONE:不使用識別碼CLASS:使用完全限定類名做識別碼MINIMAL_CLASS:使用類名(忽略包名)做識別碼,和基類在同一個(gè)包可用NAME:指定名稱CUSTOM:自定義識別碼,由@JsonTypeIdResolver對應(yīng)
- include(可選):指定識別碼如何被包含進(jìn)去,有以下幾個(gè)可選項(xiàng)。
PROPERTY:作為兄弟屬性加入,默認(rèn)值WRAPPER_OBJECT:作為一個(gè)包裝的對象WRAPPER_ARRAY:作為包裝的數(shù)組EXTERNAL_PROPERTY:作為擴(kuò)展屬性EXISTING_PROPERTY:作為已存在的屬性(符合我的場景,用channel)
- property(可選):指定識別碼的屬性名稱。該屬性只有當(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í)使用的默認(rèn)類型。
- visible(可選,默認(rèn)false):屬性定義了類型標(biāo)識符是否會成為反序列化器的一部分,默認(rèn)為false,也就是說Jackson會從json內(nèi)容中刪除類型標(biāo)識再傳遞給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屬性的值,對比以上代碼即: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請求發(fā)現(xiàn)入?yún)㈩愋陀辛俗兓?/p>


include屬性使用默認(rèn)的PROPERTY時(shí)發(fā)現(xiàn)序列化之后的json會多出來一個(gè)屬性,屬性名對應(yīng)的就是@JsonTypeInfo的property的值。雖然不影響使用,但是我看著很不舒服。基于我這種情況可以使用include=EXISTING_PROPERTY。

總結(jié)
到此這篇關(guān)于Jackson多態(tài)序列化的文章就介紹到這了,更多相關(guān)Jackson多態(tài)序列化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用正則表達(dá)式判斷獨(dú)立字符的存在(代碼示例)
通過使用正則表達(dá)式,我們可以更加靈活地判斷字符串中是否包含特定的字符,并且可以控制匹配的條件,如獨(dú)立的字符,這為我們處理字符串提供了更多的選擇和功能,這篇文章主要介紹了Java使用正則表達(dá)式判斷獨(dú)立字符的存在,需要的朋友可以參考下2023-10-10
Java?數(shù)據(jù)結(jié)構(gòu)深入理解ArrayList與順序表
ArrayList?類是一個(gè)可以動態(tài)修改的數(shù)組,與普通數(shù)組的區(qū)別就是它是沒有固定大小的限制,我們可以添加或刪除元素。ArrayList?繼承了?AbstractList?,并實(shí)現(xiàn)了?List?接口,順序表是將元素順序地存放在一塊連續(xù)的存儲區(qū)里,元素間的順序關(guān)系由它們的存儲順序自然表示2022-04-04
Mybatis傳入List實(shí)現(xiàn)批量更新的示例代碼
這篇文章主要介紹了Mybatis傳入List實(shí)現(xiàn)批量更新的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
SpringBoot中的@ControllerAdvice注解原理詳解
這篇文章主要介紹了SpringBoot中的@ControllerAdvice注解原理詳解,在SpringBoot應(yīng)用程序啟動過程中,Spring會掃描所有的類,尋找?guī)в蠤ControllerAdvice注解的類這些方法會被添加到一個(gè)映射表中,以便后續(xù)處理異常時(shí)能找到對應(yīng)的處理方法,需要的朋友可以參考下2024-01-01
springBoot啟動時(shí)讓方法自動執(zhí)行的幾種實(shí)現(xiàn)方式
這篇文章主要介紹了springBoot啟動時(shí)讓方法自動執(zhí)行的幾種實(shí)現(xiàn)方式,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03
Java基于rest assured實(shí)現(xiàn)接口測試過程解析
這篇文章主要介紹了Java基于rest assured實(shí)現(xiàn)接口測試過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
spring+mybatis實(shí)現(xiàn)圖書管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了spring+mybatis實(shí)現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-06-06

