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