乐趣区

关于java:Jackson中请求参数中枚举enum类型字段反序列化实践

背景

前后端接口交互中,常常会呈现一些枚举值,比方订单状态。前端申请参数个别是整数值,相应地,后端接管参数也用 int 类型,而后再转换成对应的枚举类型。

每次都须要手动转换未免有点麻烦,有没有什么方法能够让接管到的 int 值主动转换成我想要的枚举值呢?

实现

定义一个接口,示意能够用 int 获取 code 的枚举类型:

public interface CodeEnum {int getCode();
}

定义枚举类,实现这个接口:

public enum Direction implements CodeEnum {NORTH(0),
    EAST(1),
    SOUTH(2),
    WEST(3);

    Direction(int code) {this.code = code;}

    private final int code;

    @Override
    public int getCode() {return code;}
}

定义一个反序列化类,继承 JsonDeserializer、实现 ContextualDeserializer 接口

JsonDeserializer:用于自定义反序列化器,将 JSON 的字段值反序列化成咱们想要的类型,在这里咱们须要把 int 示意的枚举值反序列化成具体枚举类的枚举值

ContextualDeserializer:用于获取反序列字段的类型

@Slf4j
public class CodeEnumDeserializer extends JsonDeserializer<CodeEnum> implements ContextualDeserializer {

    private Class<? extends CodeEnum> propertyClass; // 记录枚举字段的类,用于获取其定义的所有枚举值

    public CodeEnumDeserializer() {log.info("Construct with no args"); // 必须有无参结构器,spring 会调用
    }

    public CodeEnumDeserializer(Class<? extends CodeEnum> propertyClass) {this.propertyClass = propertyClass;}

    @Override
    public CodeEnum deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JacksonException {
        int code;
        try {code = p.getIntValue();
        } catch (JsonParseException e) {return null;}
        log.info("Enum code: {}", code);
        return Arrays.stream(propertyClass.getEnumConstants()) // 调用 Class 的这个办法,获取枚举类的所有枚举值
                .filter(e -> e.getCode() == code)
                .findAny()
                .orElseThrow(() -> new IllegalArgumentException("No such code of" + propertyClass.getSimpleName()));
    }

    @SuppressWarnings({"unchecked"})
    @Override
    public JsonDeserializer<?> createContextual(DeserializationContext ctxt, BeanProperty property) throws
            JsonMappingException {log.info("Construct with property class");
        return new CodeEnumDeserializer((Class<? extends CodeEnum>) property.getType().getRawClass()); // 获取枚举字段的类型 Class
    }

}

接管申请的类定义:

@Data
public class TestReq {

    private String name;

    @JsonDeserialize(using = CodeEnumDeserializer.class) // 给字段指定定义的反序列化器
    private Direction direction;

}

也能够利用 @JsonComponent 注解配置成全局的,只须要在 CodeEnumDeserializer 类上加@JsonComponent(参考:Spring Boot 中应用 @JsonComponent)

退出移动版