java:jackson 五:Custom Jackson Annotation

java:jackson 五:Custom Jackson Annotation

1 前言

参考文档地址:

https://www.baeldung.com/jackson

https://www.baeldung.com/jackson-annotations

SpringBoot自带的jackson版本如下:

<parent>
    <artifactId>spring-boot-starter-parent</artifactId>
    <groupId>org.springframework.boot</groupId>
    <version>2.5.4</version>
</parent>

在这里插入图片描述

或者不使用SpringBoot情况下,单独配置jackson-databind版本:

<properties>
    <maven.compiler.source>8</maven.compiler.source>
    <maven.compiler.target>8</maven.compiler.target>
    <jackson-databind.version>2.14.1</jackson-databind.version>
</properties>
    
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>${jackson-databind.version}</version>
</dependency>

同时使用SpringBoot和自定义的jackson版本时,注意避免依赖冲突。

2 使用


@JacksonAnnotationsInside

Next let’s see how to create a custom Jackson annotation. We can make use of the @JacksonAnnotationsInside annotation

自定义jackson的注解时,如果使用了@JsonSerialize,相应的,@JsonPropertyOrder、@JsonInclude将失效,依赖自定义的序列化器等实现即可:

package com.xiaoxu.test.jackson;

import com.fasterxml.jackson.annotation.JacksonAnnotationsInside;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonPropertyOrder;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.*;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.fasterxml.jackson.databind.deser.ContextualDeserializer;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.ContextualSerializer;
import lombok.Data;
import lombok.ToString;

import java.io.IOException;
import java.lang.annotation.*;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Date;
import java.util.Optional;

/**
 * @author xiaoxu
 * @date 2022-12-30
 * java_demo:com.xiaoxu.test.jackson.CustomJackSonAnnotationTest
 */
public class CustomJackSonAnnotationTest {
    private static final ObjectMapper objectMapper;
    static {
        objectMapper = new ObjectMapper();
        objectMapper.configure(SerializationFeature.INDENT_OUTPUT,true);

        /* 单独使用了@JsonSerialize(using = CusAnnoSerializer.class),可省略全局的配置 */
        SimpleModule simpleModule = new SimpleModule();
        simpleModule.addSerializer(CusAnno.class, new CusAnnoSerializer());
        simpleModule.addDeserializer(CusAnno.class, new CusAnnoDeSerializer());
        objectMapper.registerModule(simpleModule);
    }

    public static void demo() throws Exception{
        CusAnno cusAnno = new CusAnno();
        cusAnno.setToday(new Date());
        cusAnno.setTodayNew(LocalDateTime.now());

        Optional.ofNullable(CusAnno.class.getAnnotation(CusJack.class)).ifPresent(annotation -> {
            if(annotation.isEffect()){
                System.out.println("CusJack注解生效:");
                try {
                    System.out.println(objectMapper.writeValueAsString(cusAnno));
                } catch (JsonProcessingException e) {
                    throw new RuntimeException(e);
                }
            }else{
                System.out.println("CusJack注解未生效.");
            }
        });
        System.out.print("\n\n");
        WrapCusAnno wrapCusAnno = new WrapCusAnno();
        wrapCusAnno.setMid(999);
        wrapCusAnno.setCusAnno(cusAnno);
        System.out.println(objectMapper.writeValueAsString(wrapCusAnno));

        System.out.print("\n\n");
        System.out.println(objectMapper.readValue(objectMapper.writeValueAsString(cusAnno), CusAnno.class));
        System.out.print("\n\n");
        System.out.println(objectMapper.readValue(objectMapper.writeValueAsString(wrapCusAnno), WrapCusAnno.class));
    }

    public static void main(String[] args) throws Exception{
        demo();
    }
}

class CusAnnoSerializer extends JsonSerializer<CusAnno> implements ContextualSerializer{

    @Override
    public void serialize(CusAnno cusAnno, JsonGenerator jsonGenerator, 
                          SerializerProvider serializerProvider) throws IOException {
        /* 注意:若使用writeObject()方法,不要直接把参数 cusAnno 再次写入,会造成StackOverflowError 错误*/
        jsonGenerator.writeStartObject();
        /* 改变自定义的顺序即可实现类似 @JsonPropertyOrder(value = {"today","todayNew","name","id"}) 的效果 */
        jsonGenerator.writeStringField("name",cusAnno.getName());
        jsonGenerator.writeNumberField("id",cusAnno.id+1);

        Optional.ofNullable(cusAnno.getToday()).ifPresent(date -> {
            try {
                jsonGenerator.writeStringField("nowDay",cusAnno.getToday().toString());
            } catch (IOException e) {
                throw new RuntimeException("today转换异常:"+e.getMessage(),e.getCause());
            }
        });
        Optional.ofNullable(cusAnno.getTodayNew()).ifPresent(localDate -> {
            try {
                jsonGenerator.writeStringField("localDate",
                        cusAnno.getTodayNew().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm")));
            } catch (IOException e) {
                throw new RuntimeException("todayNew转换异常:"+e.getMessage(),e.getCause());
            }
        });
        jsonGenerator.writeEndObject();
    }

    @Override
    public JsonSerializer<CusAnno> createContextual(SerializerProvider serializerProvider, 
                                                    BeanProperty beanProperty) throws JsonMappingException {
        System.out.println("序列化BeanProperty:"+beanProperty);
        Optional.ofNullable(beanProperty).ifPresent(beanProperty1 -> {
            System.out.println("序列化BeanProperty不为null:"+beanProperty1.getAnnotation(CusJack.class));
        });
        return this;
    }

}

class CusAnnoDeSerializer extends JsonDeserializer<CusAnno> implements ContextualDeserializer{

    @Override
    public CusAnno deserialize(JsonParser jsonParser, 
                               DeserializationContext deserializationContext) 
            throws IOException, JsonProcessingException {
        JsonNode node = jsonParser.getCodec().readTree(jsonParser);
        int id = node.get("id").intValue();
        String name = node.get("name").asText();
        /* 注意:node.get("name"),若name不存在,则该值为null,可能导致NPE */
        if(node.get("name").isNull()){
            name = "defualt-xiaoxu";
        }

        String localDate = node.get("localDate").asText();
        LocalDateTime dtime = LocalDateTime.parse(localDate, 
                DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm"));

        CusAnno cusAnnoDe = new CusAnno();
        cusAnnoDe.setId(id);
        cusAnnoDe.setName(name);
        cusAnnoDe.setToday(new Date());
        cusAnnoDe.setTodayNew(dtime);
        return cusAnnoDe;
    }

    @Override
    public JsonDeserializer<?> createContextual(DeserializationContext deserializationContext, 
                                                BeanProperty beanProperty) throws JsonMappingException {
        System.out.println("反序列化BeanProperty:"+beanProperty);
        Optional.ofNullable(beanProperty).ifPresent(beanProperty1 -> {
            System.out.println("反序列化BeanProperty不为null:"+beanProperty1.getAnnotation(CusJack.class));
        });
        return this;
    }
}

//@CusJack  类上一定不能有该注解
@ToString
@Data
class WrapCusAnno{
    int mid;
    //字段上可有可无该@CusJack注解,但是类上不能有,直接带上即可
    @CusJack
    CusAnno cusAnno;
}

@CusJack(isEffect = true)
@Data
class CusAnno{
    int id;
    String name;
    Date today;
    LocalDateTime todayNew;
}

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.FIELD})
@JacksonAnnotationsInside
/* 使用 JsonSerialize自定义的序列化器时,序列化注解@JsonPropertyOrder、
@JsonInclude将不会生效,需要自定义展示效果 */
@JsonPropertyOrder(value = {"today","todayNew","name","id"})
@JsonInclude(JsonInclude.Include.NON_NULL)
@JsonSerialize(using = CusAnnoSerializer.class)
@JsonDeserialize(using = CusAnnoDeSerializer.class)
@interface CusJack{
    boolean isEffect() default false;
}

执行结果如下:

CusJack注解生效:
序列化BeanProperty:null
{
  "name" : null,
  "id" : 1,
  "nowDay" : "Fri Dec 30 16:33:18 CST 2022",
  "localDate" : "2022-12-30 16:33"
}


序列化BeanProperty:com.fasterxml.jackson.databind.BeanProperty$Std@67205a84
序列化BeanProperty不为null:@com.xiaoxu.test.jackson.CusJack(isEffect=false)
{
  "mid" : 999,
  "cusAnno" : {
    "name" : null,
    "id" : 1,
    "nowDay" : "Fri Dec 30 16:33:18 CST 2022",
    "localDate" : "2022-12-30 16:33"
  }
}


反序列化BeanProperty:null
CusAnno(id=1, name=defualt-xiaoxu, today=Fri Dec 30 16:33:18 CST 2022, todayNew=2022-12-30T16:33)


反序列化BeanProperty:[property 'cusAnno']
反序列化BeanProperty不为null:@com.xiaoxu.test.jackson.CusJack(isEffect=false)
反序列化BeanProperty:[property 'cusAnno']
反序列化BeanProperty不为null:@com.xiaoxu.test.jackson.CusJack(isEffect=false)
WrapCusAnno(mid=999, cusAnno=CusAnno(id=1, name=defualt-xiaoxu, today=Fri Dec 30 16:33:18 CST 2022, todayNew=2022-12-30T16:33))

可见,自定义jackson注解,可以整合多个注解,使用自定义的1个注解,可达到相同效果。


### 如何在Java中正确解析和生成JSON日期格式 #### 使用Gson库处理JSON日期 当使用Gson库来处理带有日期字段的对象时,可以通过自定义`TypeAdapter`或设置日期格式模式来确保日期被正确地序列化和反序列化。 对于日期类型的属性,默认情况下Gson会尝试按照ISO 8601标准格式(如:"yyyy-MM-dd'T'HH:mm:ss.SSSZ")进行转换。如果希望指定其他格式,则需创建并注册一个适配器[^1]。 ```java import com.google.gson.Gson; import com.google.gson.GsonBuilder; import java.util.Date; public class DateExample { static class Event { public String name; public Date eventDate; // Use the Date type for date fields public Event(String name, Date eventDate) { this.name = name; this.eventDate = eventDate; } } public static void main(String[] args) throws Exception{ Gson gson = new GsonBuilder() .setDateFormat("yyyy-MM-dd") // Set custom date format here .create(); Event event = new Event("Birthday Party", new Date()); // Convert object to JSON string with specified date format String jsonOutput = gson.toJson(event); System.out.println(jsonOutput); // Parse from JSON back into an object using defined formatter Event parsedEvent = gson.fromJson(jsonOutput, Event.class); System.out.println(parsedEvent.eventDate); } } ``` 此代码片段展示了如何通过`GsonBuilder().setDateFormat()`方法设定全局日期格式,从而影响所有`Date`类型的序列化/反序列化进程[^2]。 另外,在更复杂的应用场景下,可能还需要考虑时间区域的支持以及不同的输入输出需求。此时可以进一步扩展功能,比如利用`SimpleDateFormat`类来自定义更加灵活的时间表示方式,并将其集成到Gson配置当中[^3]。 #### 使用Jackson库处理JSON日期 除了Gson之外,Jackson也是一个非常流行的用于JSON处理的框架。它提供了更为丰富的API接口来进行高级定制化的操作,包括但不限于对日期的支持: ```java import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.annotation.JsonFormat; import java.text.SimpleDateFormat; import java.util.Date; class Person { private String firstName; @JsonFormat(shape= JsonFormat.Shape.STRING, pattern="dd/MM/yyyy") private Date birthdate; // Getters and setters... } // ... ObjectMapper mapper = new ObjectMapper(); mapper.setDateFormat(new SimpleDateFormat("MM/dd/yyyy")); Person user = new Person(); user.setFirstName("John"); user.setBirthdate(new Date()); String jsonString = mapper.writeValueAsString(user); System.out.println(jsonString); ``` 上述例子说明了怎样借助于注解(`@JsonFormat`)的方式直接在实体类内部声明特定的数据交换格式;同时也演示了如何在整个应用范围内统一调整默认的行为——即通过修改`ObjectMapper`实例中的日期格式设置[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值