[英]How do you modify default enum (de)serialization for non-annotated enums but retain standard behavior (@JsonProperty/@JsonValue/...) in Jackson?
目前 jackson (uncustomized) 像這樣序列化枚舉:
@JsonProperty
或@JsonValue
,它們用於派生序列化名稱toString()
(取決於映射器設置),或默認調用.name()
對於反序列化,是這樣的:
@JsonProperty
、 @JsonValue
或@JsonCreator
,它們會影響反序列化 我想在這兩種情況下都保留第 1 項(即仍然支持所有 3 個注釋,如果我遺漏了一些注釋,或者更多),但修改“默認情況”的行為:例如,始終將枚舉序列化為.name().toLowercase()
如果沒有注釋。
我可以使用addSerializer()
、 addDeserializer()
或 (de)serializer 修改機制,但這些都不允許簡單而優雅的解決方案。 我所能發明的只是從 jackson 復制/粘貼大量代碼(這是丑陋和脆弱的),或者,使用修改機制,內省模擬 jackson 涉及的邏輯的枚舉類,以識別“默認”邏輯何時適用的情況,然后使用我的“小寫”策略。
有沒有更好的方法來修改“默認”序列化策略,同時仍然保留所有“非默認”情況?
您可以插入一個新的AnnotationIntrospector,其findEnumValues
方法會更改枚舉名稱。 該EnumRenamingModule
從therapi-JSON-RPC項目使用了這種技術。 (免責聲明:這是我的寵物項目。)這是從 GitHub 存儲庫復制的代碼:
package com.github.therapi.jackson.enums;
import com.fasterxml.jackson.databind.Module;
import com.fasterxml.jackson.databind.introspect.NopAnnotationIntrospector;
import com.fasterxml.jackson.databind.module.SimpleModule;
/**
* Customizes the way Jackson serializes enums.
*/
public abstract class EnumRenamingModule extends SimpleModule {
private boolean overrideExistingNames;
public EnumRenamingModule() {
super("therapi-enum-renaming");
}
/**
* Configures the module to clobber any enum names set by
* a previous annotation introspector.
*/
public EnumRenamingModule overrideExistingNames() {
this.overrideExistingNames = true;
return this;
}
@Override
public void setupModule(Module.SetupContext context) {
super.setupModule(context);
context.insertAnnotationIntrospector(new EnumNamingAnnotationIntrospector());
}
private class EnumNamingAnnotationIntrospector extends NopAnnotationIntrospector {
public String[] findEnumValues(Class<?> enumType, Enum<?>[] enumValues, String[] names) {
for (int i = 0; i < enumValues.length; i++) {
if (names[i] == null || overrideExistingNames) {
names[i] = EnumRenamingModule.this.getName(enumValues[i]);
}
}
return names;
}
}
/**
* @param value the enum value to inspect
* @return the JSON name for the enum value, or {@code null} to delegate to the next introspector
*/
protected abstract String getName(Enum<?> value);
}
創建一個子類來做你想要的轉換:
public class LowerCaseEnumModule extends EnumRenamingModule {
@Override protected String getName(Enum<?> value) {
return value.name().toLowerCase(Locale.ROOT);
}
}
並使用您的 ObjectMapper 注冊它:
ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new LowerCaseEnumModule());
注解@JsonCreator
用於反序列化(它是一個靜態工廠)。
注解@JsonValue
用於序列化。
給定的例子對我有用:
class MyException extends RuntimeException {}
enum MyEnum {
TEST;
private final String value;
private MyEnum() {
this.value = this.name().toLowerCase();
}
@JsonValue
public final String value() {
return this.value;
}
@JsonCreator
public static MyEnum forValue(String value) {
return Arrays.stream(MyEnum.values()).filter(myEnum -> myEnum.value.equals(value)).findFirst().orElseThrow(MyException::new);
}
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.