Jackson處理Optional時(shí)遇到問題的解決與分析
前言
Optional是Java8中增加的一個(gè)特性,它的出現(xiàn)是為了解決Java中的空指針問題,相關(guān)介紹可以參考這篇 Java8中的Optional操作;
但是在Jackson中操作Optional類型的屬性時(shí),會(huì)遇到一些問題,比如序列化的數(shù)據(jù)不符合預(yù)期等;
下面就來介紹下遇到的問題以及如何解決;
目錄
- 序列化Optional類型的問題
- 原因分析
- 解決辦法
正文
1. 序列化Optional類型的問題
其他類型的屬性序列化時(shí)基本沒啥問題,都會(huì)根據(jù)對(duì)象的值進(jìn)行序列化;
但是Optional比較特殊,序列化時(shí)會(huì)輸出present:true這樣的數(shù)據(jù);
下面我們看下例子;
這是User對(duì)象,其中nickname為Optional類型:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
public String username;
public Optional<String> nickname;
}序列化的代碼如下所示:
User user = new User("jalon", Optional.of("xiaowang"));
ObjectMapper objectMapper = new ObjectMapper();
String str = objectMapper.writeValueAsString(user);
System.out.println(str);這里預(yù)期的結(jié)果應(yīng)該是類似下面這樣的:
{"username":"jalon","nickname":"xiaowang"}但實(shí)際輸出如下所示:

2. 原因分析
之所以序列化會(huì)輸出{"present":true}這樣的字符串,是因?yàn)镴ackson默認(rèn)的序列化行為導(dǎo)致;
Jackson默認(rèn)的序列化會(huì)把所有public類型的get方法進(jìn)行序列化,也就是取出對(duì)象中所有可訪問的屬性,然后填充到結(jié)果中;
而這里的Optional對(duì)象默認(rèn)只有一個(gè)public類型的get方法,就是isPresent(),這個(gè)方法會(huì)返回true(當(dāng)Optional的值不為空)或者false(當(dāng)Optional的值為空);
Optional類的局部?jī)?nèi)容如下所示:
public final class Optional<T> {
private final T value;
public T get() {
if (value == null) {
throw new NoSuchElementException("No value present");
}
return value;
}
public boolean isPresent() {
return value != null;
}
@Override
public String toString() {
return value != null
? String.format("Optional[%s]", value)
: "Optional.empty";
}
}可以看到,雖然有一個(gè)value屬性,但因?yàn)槭莗rivate類型,所以無法直接被Jackson讀取;
所以此時(shí)Jackson默認(rèn)只讀取了isPresent()方法,取得了true值;
3. 解決辦法
幸運(yùn)的是,Jackson官方已經(jīng)出了一個(gè)maven依賴,專門用來解決 由于Java8新增的數(shù)據(jù)類型導(dǎo)致的各種問題;
添加如下依賴:
<dependency>
<groupId>com.fasterxml.jackson.datatype</groupId>
<artifactId>jackson-datatype-jdk8</artifactId>
<version>2.12.5</version>
</dependency>然后在ObjectMapper對(duì)象中配置jdk8模塊:objectMapper.registerModule(new Jdk8Module());
User user = new User("jalon", Optional.of("xiaowang"));
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.registerModule(new Jdk8Module());
String str = objectMapper.writeValueAsString(user);
System.out.println(str);最后輸出符合預(yù)期,如下所示:

總結(jié)
Jackson在操作Optional類型的屬性時(shí),會(huì)由于Jackson自身的默認(rèn)行為,導(dǎo)致輸出的結(jié)果不符合預(yù)期;
解決辦法就是加載jackson-datatype-jdk8依賴,然后全局注冊(cè)Java8模塊Jdk8Module;
到此這篇關(guān)于Jackson處理Optional時(shí)遇到問題的解決與分析的文章就介紹到這了,更多相關(guān)Jackson處理Optional內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot整合Redis之編寫RedisConfig
RedisConfig需要對(duì)redis提供的兩個(gè)Template的序列化配置,所以本文為大家詳細(xì)介紹了SpringBoot整合Redis如何編寫RedisConfig,需要的可以參考下2022-06-06
spring-boot-starter-parent的作用詳解
這篇文章主要介紹了spring-boot-starter-parent的作用詳解,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08
springboot實(shí)現(xiàn)rabbitmq消息確認(rèn)的示例代碼
RabbitMQ的消息確認(rèn)有兩種, 一種是消息發(fā)送確認(rèn),第二種是消費(fèi)接收確認(rèn),本文主要介紹了springboot實(shí)現(xiàn)rabbitmq消息確認(rèn)的示例代碼,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09
使用Java的Spring框架編寫第一個(gè)程序Hellow world
這篇文章主要介紹了Java的Spring框架并用其開始編寫第一個(gè)程序Hellow world的方法,Spring是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下2015-12-12
教你如何編寫簡(jiǎn)單的網(wǎng)絡(luò)爬蟲
實(shí)際的爬蟲是從一系列的種子鏈接開始。種子鏈接是起始節(jié)點(diǎn),種子頁面的超鏈接指向的頁面是子節(jié)點(diǎn)(中間節(jié)點(diǎn)),對(duì)于非html文檔,如excel等,不能從中提取超鏈接,看做圖的終端節(jié)點(diǎn)2013-10-10
SpringMVC實(shí)現(xiàn)獲取請(qǐng)求參數(shù)方法詳解
Spring MVC 是 Spring 提供的一個(gè)基于 MVC 設(shè)計(jì)模式的輕量級(jí) Web 開發(fā)框架,本質(zhì)上相當(dāng)于 Servlet,Spring MVC 角色劃分清晰,分工明細(xì),這篇文章主要介紹了SpringMVC實(shí)現(xiàn)獲取請(qǐng)求參數(shù)方法2022-09-09
SpringBoot修改內(nèi)置tomcat版本的操作步驟
生產(chǎn)環(huán)境使用的外部部署Tomcat還是內(nèi)置Tomcat由于版本安全漏洞,往往需要升級(jí)到指定的安全版本,本文演示一下SpringBoot升級(jí)內(nèi)置的Tomcat版本,感興趣的小伙伴跟著小編一起來看看吧2024-07-07

