溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

SpringBoot之Json的序列化和反序列化問(wèn)題怎么解決

發(fā)布時(shí)間:2022-06-15 13:50:16 來(lái)源:億速云 閱讀:447 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“SpringBoot之Json的序列化和反序列化問(wèn)題怎么解決”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“SpringBoot之Json的序列化和反序列化問(wèn)題怎么解決”吧!

控制json序列化/反序列化

1. @JsonIgnoreProperties的用法

@JsonIgnoreProperties(value = { "prop1","prop2" })

用來(lái)修飾Pojo類, 在序列化和反序列化的時(shí)候忽略指定的屬性, 可以忽略一個(gè)或多個(gè)屬性.

@JsonIgnoreProperties(ignoreUnknown = true)

用來(lái)修飾Pojo類, 在反序列化的時(shí)候忽略那些無(wú)法被設(shè)置的屬性, 包括無(wú)法在構(gòu)造子設(shè)置和沒有對(duì)應(yīng)的setter方法.

2. @JsonProperty 注解

如果 json field 的名稱和Pojo 的屬性名不一致的時(shí), 可以用 @JsonProperty 來(lái)注解 getter() 或 setter() 方法, 該注解設(shè)定json 對(duì)應(yīng)的屬性名, 另外@JsonProperty也經(jīng)常用來(lái)注解構(gòu)造子的形參, 這時(shí)候構(gòu)造子應(yīng)該加@JsonCreator 注解.

3. @JsonCreator 注解

如果 Pojo 類定義有參數(shù)的構(gòu)造子, 但沒有提供無(wú)參構(gòu)造子時(shí), 在反序列化時(shí)是會(huì)報(bào)錯(cuò). 有下面兩個(gè)辦法:

  • 方法1:增加一個(gè)無(wú)參構(gòu)造子

  • 方法2:為這個(gè)有參數(shù)的構(gòu)造子, 加上 @JsonCreator 注解, 同時(shí)參數(shù)需要加上 @JsonProperty 注解.

4. @JsonSetter 和 @JsonGetter 注解

如果 json field 的名稱和Pojo 的屬性名不一致的時(shí), 可以使用@JsonGetter來(lái)注解 getter(), 使用 @JsonSetter 來(lái)注解setter() 方法. 這兩個(gè)注解都可以指定一個(gè)屬性名. 這兩個(gè)注解都可以用 @JsonProperty 替換.

5. @JsonAnySetter 注解

一般對(duì)象屬性名都是確定的, 比如 Car 這個(gè)對(duì)象, 有 brand/price 等具名屬性, 但有時(shí)候我們還需要為Car這個(gè)對(duì)象設(shè)定一些擴(kuò)展屬性, 這些擴(kuò)展屬性名稱暫時(shí)不好確定, 通常使用 Map<String, String> 來(lái)存放這些擴(kuò)展屬性的K/V. 要把json 數(shù)據(jù)中的這些屬性反序列化到類的Map中, 需要在類上增加一個(gè)K/V的setter方法, 而且這個(gè)setter方法要加上@JsonAnySetter注解.

public class Car {
    public String brand;
    private Map<String, String> properties;
 
    @JsonAnySetter
    public void add(String key, String value) {
        properties.put(key, value);
    }
}

一旦類加上了@JsonAnySetter后可將下面的 json 數(shù)據(jù)反序列化到Map中:

{
"brand":"Benz",
"attr2":"val2",
"attr1":"val1"
}

6. @JsonAnyGetter 注解

和@JsonAnySetter 注解相對(duì)應(yīng), 如果要將類中的Map K/V屬性序列化到j(luò)son中, 需要在類上增加一個(gè) @JsonAnyGetter 方法, 該方法直接返回KV map就行.

public class Car {
    public String brand;
    private Map<String, String> properties;
 
    @JsonAnyGetter
    public Map<String, String> getProperties() {
        return properties;
    }
}

序列化后的json為:

{
"brand":"Benz",
"attr2":"val2",
"attr1":"val1"
}

7. @JsonFormat 注解

通常明確Date/Time 屬性序列化用的時(shí)間格式.

public class Event {
    public String name;
 
    @JsonFormat(
      shape = JsonFormat.Shape.STRING, pattern = "dd-MM-yyyy hh:mm:ss")
    public Date eventDate;
}

8.@JsonSerialize 和 @JsonDeserialize 注解

@JsonSerialize注解可以為類屬性設(shè)定專門的序列化函數(shù), @JsonDeserialize注解用來(lái)為json屬性定制化反序列化函數(shù)

SpringBoot序列化規(guī)則

  • Include.Include.ALWAYS 默認(rèn)

  • Include.NON_DEFAULT 屬性為默認(rèn)值不序列化

  • Include.NON_EMPTY 屬性為 空(“”) 或者為 NULL 都不序列化

  • Include.NON_NULL 屬性為NULL 不序列化

1.全局設(shè)置

/**
 * 〈返回json空值去掉null和""〉 〈功能詳細(xì)描述〉
 * 
 * @author gogym
 * @version 2017年10月13日
 * @see JacksonConfig
 * @since
 */
@Configuration
public class JacksonConfig
{
    @Bean
    @Primary
    @ConditionalOnMissingBean(ObjectMapper.class)
    public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder)
    {
        ObjectMapper objectMapper = builder.createXmlMapper(false).build();
        // 通過(guò)該方法對(duì)mapper對(duì)象進(jìn)行設(shè)置,所有序列化的對(duì)象都將按改規(guī)則進(jìn)行系列化
        // Include.Include.ALWAYS 默認(rèn)
        // Include.NON_DEFAULT 屬性為默認(rèn)值不序列化
        // Include.NON_EMPTY 屬性為 空("") 或者為 NULL 都不序列化,則返回的json是沒有這個(gè)字段的。這樣對(duì)移動(dòng)端會(huì)更省流量
        // Include.NON_NULL 屬性為NULL 不序列化,就是為null的字段不參加序列化
        //objectMapper.setSerializationInclusion(Include.NON_EMPTY);
        objectMapper.setSerializationInclusion(JsonInclude.Include.NON_EMPTY);
        return objectMapper;
    }
}

或者

spring.jackson.default-property-inclusion=non_null
spring:
  jackson:
    default-property-inclusion: non_null

2.局部設(shè)置

在需要設(shè)置的實(shí)體類或者字段上加上注解

@JsonInclude(Include.NON_NULL)

3.自定義局部序列化

(1)、自定義一個(gè)序列化工具類,需要實(shí)現(xiàn)StdSerializer<T>或者JsonSerializer<T>

public class ClientObjectSerialize extends JsonSerializer<CreditBorrowerRepaymentRequestDto>{
 @Override
 public void serialize(CreditBorrowerRepaymentRequestDto dto, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException, JsonProcessingException {
 
  jsonGenerator.writeStartObject();
  try {
   Field[] fields = dto.getClass().getDeclaredFields();
   for (Field field : fields) {
    field.setAccessible(true);
    if(null == field.get(dto)){
     continue; 
    }
    jsonGenerator.writeFieldName(field.getName());
    jsonGenerator.writeObject(field.get(dto));
   }
  } catch (Exception e) {
   e.printStackTrace();
  }
  jsonGenerator.writeEndObject();
 }
}

(2)使用注解作用在實(shí)體類上

@JsonSerialize(using = ClientObjectSerialize.class)
public class CreditBorrowerRepaymentRequestDto{
}

(3)可以作用在實(shí)體對(duì)象字段上,對(duì)NULL值的處理,或者轉(zhuǎn)換

@JsonSerialize(using = ClientStringSerialize.class)
private String name;
 
@JsonSerialize(using = ClientDtaeSerialize.class)
private Date date;
public class ClientStringSerialize extends JsonSerializer<String> {
 
 @Override
 public void serialize(String string, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException {
 
  if(string == null){
   jsonGenerator.writeString(string + "[NULL]");
  }else{
   jsonGenerator.writeString(string);
  }
 }
}
 
public class ClientDtaeSerialize extends JsonSerializer<Date> {
 @Override
 public void serialize(Date createDate, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException {
  jsonGenerator.writeString(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(createDate));
 }
}

4.自定義全局 null 轉(zhuǎn)換 序列化

SpringBoot返回Json數(shù)據(jù)中null值處理,將字符串類型null值轉(zhuǎn)換為"",將集合數(shù)組類型null值轉(zhuǎn)換為[],將原始數(shù)據(jù)類型null值轉(zhuǎn)換為0,將布爾類型null值轉(zhuǎn)換為false,將實(shí)體對(duì)象null值轉(zhuǎn)換為{}。

(1)自定義null值序列化處理器

import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.SerializerProvider;
import java.io.IOException;
/**
 * 自定義null值序列化處理器
 */
public class CustomizeNullJsonSerializer {
 
 /**
  * 處理數(shù)組集合類型的null值
  */
 public static class NullArrayJsonSerializer extends JsonSerializer<Object> {
  @Override
  public void serialize(Object value, JsonGenerator jsonGenerator, 
    SerializerProvider serializerProvider) throws IOException {
   jsonGenerator.writeStartArray();
   jsonGenerator.writeEndArray();
  }
 }
 
 /**
  * 處理字符串類型的null值
  */
 public static class NullStringJsonSerializer extends JsonSerializer<Object> {
  @Override
  public void serialize(Object value, JsonGenerator jsonGenerator, 
    SerializerProvider serializerProvider) throws IOException {
   jsonGenerator.writeString("");
  }
 }
 
 /**
  * 處理數(shù)值類型的null值
  */
 public static class NullNumberJsonSerializer extends JsonSerializer<Object> {
  @Override
  public void serialize(Object value, JsonGenerator jsonGenerator, 
        SerializerProvider serializerProvider) throws IOException {
   jsonGenerator.writeNumber(0);
  }
 }
 
 /**
  * 處理boolean類型的null值
  */
 public static class NullBooleanJsonSerializer extends JsonSerializer<Object> {
  @Override
  public void serialize(Object value, JsonGenerator jsonGenerator, 
        SerializerProvider serializerProvider) throws IOException {
   jsonGenerator.writeBoolean(false);
  }
 }
 
 /**
  * 處理實(shí)體對(duì)象類型的null值
  */
 public static class NullObjectJsonSerializer extends JsonSerializer<Object> {
  @Override
  public void serialize(Object value, JsonGenerator jsonGenerator, 
        SerializerProvider serializerProvider) throws IOException {
   jsonGenerator.writeStartObject();
   jsonGenerator.writeEndObject();
  }
 }
}

(2)序列化程序修改器

import com.fasterxml.jackson.databind.BeanDescription;
import com.fasterxml.jackson.databind.SerializationConfig;
import com.fasterxml.jackson.databind.ser.BeanPropertyWriter;
import com.fasterxml.jackson.databind.ser.BeanSerializerModifier;
import java.util.Collection;
import java.util.List;
 
/**
 * <pre>
 * 此modifier主要做的事情為:
 * 1.當(dāng)序列化類型為數(shù)組集合時(shí),當(dāng)值為null時(shí),序列化成[]
 * 2.String類型值序列化為""
 *
 * </pre>
 */
public class MyBeanSerializerModifier extends BeanSerializerModifier {
 
 @Override
 public List<BeanPropertyWriter> changeProperties(SerializationConfig config, 
              BeanDescription beanDesc,
              List<BeanPropertyWriter> beanProperties) {
  // 循環(huán)所有的beanPropertyWriter
  for (int i = 0; i < beanProperties.size(); i++) {
   BeanPropertyWriter writer = beanProperties.get(i);
   // 判斷字段的類型,如果是數(shù)組或集合則注冊(cè)nullSerializer
   if (isArrayType(writer)) {
    // 給writer注冊(cè)一個(gè)自己的nullSerializer 
    writer.assignNullSerializer(new CustomizeNullJsonSerializer.NullArrayJsonSerializer());
   }
   if (isStringType(writer)) {
    writer.assignNullSerializer(new CustomizeNullJsonSerializer.NullStringJsonSerializer());
   }
  }
  return beanProperties;
 }
 
 /**
  * 是否是數(shù)組
  */
 private boolean isArrayType(BeanPropertyWriter writer) {
  Class<?> clazz = writer.getType().getRawClass();
  return clazz.isArray() || Collection.class.isAssignableFrom(clazz);
 }
 
 /**
  * 是否是String
  */
 private boolean isStringType(BeanPropertyWriter writer) {
  Class<?> clazz = writer.getType().getRawClass();
  return CharSequence.class.isAssignableFrom(clazz) || Character.class.isAssignableFrom(clazz);
 }
 
 /**
  * 是否是數(shù)值類型
  */
 private boolean isNumberType(BeanPropertyWriter writer) {
  Class<?> clazz = writer.getType().getRawClass();
  return Number.class.isAssignableFrom(clazz);
 }
 
 /**
  * 是否是boolean
  */
 private boolean isBooleanType(BeanPropertyWriter writer) {
  Class<?> clazz = writer.getType().getRawClass();
  return clazz.equals(Boolean.class);
 }
}

(3)配置Jackson實(shí)體

import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializerProvider;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 
/**
 * 配置Jackson實(shí)體
 */
@Configuration
public class JacksonConfig {
 @Bean
 @Primary
 @ConditionalOnMissingBean(ObjectMapper.class)
 public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
  ObjectMapper objectMapper = builder.createXmlMapper(false).build();
  /** 為objectMapper注冊(cè)一個(gè)帶有SerializerModifier的Factory */
  objectMapper.setSerializerFactory(objectMapper.getSerializerFactory()
    .withSerializerModifier(new MyBeanSerializerModifier()));
 
  SerializerProvider serializerProvider = objectMapper.getSerializerProvider();
  serializerProvider.setNullValueSerializer(new CustomizeNullJsonSerializer
 .NullObjectJsonSerializer());
  return objectMapper;
 }
}

感謝各位的閱讀,以上就是“SpringBoot之Json的序列化和反序列化問(wèn)題怎么解決”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)SpringBoot之Json的序列化和反序列化問(wèn)題怎么解決這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI