Mam Enum opisane poniżej:
public enum OrderType {
UNKNOWN(0, "Undefined"),
TYPEA(1, "Type A"),
TYPEB(2, "Type B"),
TYPEC(3, "Type C");
private Integer id;
private String name;
private WorkOrderType(Integer id, String name) {
this.id = id;
this.name = name;
}
//Setters, getters....
}
Zwracam tablicę wyliczeń z moim kontrolerem ( new OrderType[] {UNKNOWN,TYPEA,TYPEB,TYPEC};
), a Spring serializuje ją do następującego ciągu json:
["UNKNOWN", "TYPEA", "TYPEB", "TYPEC"]
Jakie jest najlepsze podejście, aby zmusić Jacksona do serializacji wyliczeń, tak jak POJO? Na przykład:
[
{"id": 1, "name": "Undefined"},
{"id": 2, "name": "Type A"},
{"id": 3, "name": "Type B"},
{"id": 4, "name": "Type C"}
]
Bawiłem się różnymi adnotacjami, ale nie udało mi się uzyskać takiego wyniku.
Odpowiedzi:
W końcu sam znalazłem rozwiązanie.
Musiałem dodać adnotacje do wyliczenia
@JsonSerialize(using = OrderTypeSerializer.class)
i zaimplementować niestandardowy serializator:public class OrderTypeSerializer extends JsonSerializer<OrderType> { @Override public void serialize(OrderType value, JsonGenerator generator, SerializerProvider provider) throws IOException, JsonProcessingException { generator.writeStartObject(); generator.writeFieldName("id"); generator.writeNumber(value.getId()); generator.writeFieldName("name"); generator.writeString(value.getName()); generator.writeEndObject(); } }
źródło
@JsonFormat(shape= JsonFormat.Shape.OBJECT) public enum SomeEnum
dostępne od https://github.com/FasterXML/jackson-databind/issues/24
właśnie przetestowałem, że działa z wersją 2.1.2
odpowiedź na TheZuck :
Próbowałem twojego przykładu, dostałem Json:
{"events":[{"type":"ADMIN"}]}
Mój kod:
@RequestMapping(value = "/getEvent") @ResponseBody public EventContainer getEvent() { EventContainer cont = new EventContainer(); cont.setEvents(Event.values()); return cont; } class EventContainer implements Serializable { private Event[] events; public Event[] getEvents() { return events; } public void setEvents(Event[] events) { this.events = events; } }
a zależności to:
<dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-annotations</artifactId> <version>${jackson.version}</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version>${jackson.version}</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>${jackson.version}</version> <exclusions> <exclusion> <artifactId>jackson-annotations</artifactId> <groupId>com.fasterxml.jackson.core</groupId> </exclusion> <exclusion> <artifactId>jackson-core</artifactId> <groupId>com.fasterxml.jackson.core</groupId> </exclusion> </exclusions> </dependency> <jackson.version>2.1.2</jackson.version>
źródło
@JsonFormat
wyliczenie, gdy jest używane w innej encji? na przykład jednostka, w której chcę, aby wyliczenie było serializowane jako ciąg znaków zamiast obiektu. próbuję dodać kolejny@JsonFormat
w polu w klasie, która używa wyliczenia, ale zawsze jest on serializowany jako obiekt.Znalazłem bardzo ładne i zwięzłe rozwiązanie, szczególnie przydatne, gdy nie można modyfikować klas wyliczeniowych, jak to było w moim przypadku. Następnie należy zapewnić niestandardowy ObjectMapper z włączoną określoną funkcją. Te funkcje są dostępne od wersji Jackson 1.6.
public class CustomObjectMapper extends ObjectMapper { @PostConstruct public void customConfiguration() { // Uses Enum.toString() for serialization of an Enum this.enable(WRITE_ENUMS_USING_TO_STRING); // Uses Enum.toString() for deserialization of an Enum this.enable(READ_ENUMS_USING_TO_STRING); } }
Dostępnych jest więcej funkcji związanych z wyliczeniami, zobacz tutaj:
https://github.com/FasterXML/jackson-databind/wiki/Serialization-features https://github.com/FasterXML/jackson-databind/wiki/Deserialization-Features
źródło
objMapper.enable(SerializationFeature.WRITE_ENUMS_USING_TO_STRING);
i to:objMapper.enable(DeserializationFeature.READ_ENUMS_USING_TO_STRING);
Oto moje rozwiązanie. Chcę przekształcić wyliczenie do
{id: ..., name: ...}
postaci.Z Jacksonem 1.x :
pom.xml:
<properties> <jackson.version>1.9.13</jackson.version> </properties> <dependencies> <dependency> <groupId>org.codehaus.jackson</groupId> <artifactId>jackson-core-asl</artifactId> <version>${jackson.version}</version> </dependency> <dependency> <groupId>org.codehaus.jackson</groupId> <artifactId>jackson-mapper-asl</artifactId> <version>${jackson.version}</version> </dependency> </dependencies>
Rule.java:
import org.codehaus.jackson.map.annotate.JsonSerialize; import my.NamedEnumJsonSerializer; import my.NamedEnum; @Entity @Table(name = "RULE") public class Rule { @Column(name = "STATUS", nullable = false, updatable = true) @Enumerated(EnumType.STRING) @JsonSerialize(using = NamedEnumJsonSerializer.class) private Status status; public Status getStatus() { return status; } public void setStatus(Status status) { this.status = status; } public static enum Status implements NamedEnum { OPEN("open rule"), CLOSED("closed rule"), WORKING("rule in work"); private String name; Status(String name) { this.name = name; } public String getName() { return this.name; } }; }
NamedEnum.java:
package my; public interface NamedEnum { String name(); String getName(); }
NamedEnumJsonSerializer.java:
package my; import my.NamedEnum; import java.io.IOException; import java.util.*; import org.codehaus.jackson.JsonGenerator; import org.codehaus.jackson.JsonProcessingException; import org.codehaus.jackson.map.JsonSerializer; import org.codehaus.jackson.map.SerializerProvider; public class NamedEnumJsonSerializer extends JsonSerializer<NamedEnum> { @Override public void serialize(NamedEnum value, JsonGenerator jgen, SerializerProvider provider) throws IOException, JsonProcessingException { Map<String, String> map = new HashMap<>(); map.put("id", value.name()); map.put("name", value.getName()); jgen.writeObject(map); } }
Z Jackson 2.x :
pom.xml:
<properties> <jackson.version>2.3.3</jackson.version> </properties> <dependencies> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version>${jackson.version}</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>${jackson.version}</version> </dependency> </dependencies>
Rule.java:
import com.fasterxml.jackson.annotation.JsonFormat; @Entity @Table(name = "RULE") public class Rule { @Column(name = "STATUS", nullable = false, updatable = true) @Enumerated(EnumType.STRING) private Status status; public Status getStatus() { return status; } public void setStatus(Status status) { this.status = status; } @JsonFormat(shape = JsonFormat.Shape.OBJECT) public static enum Status { OPEN("open rule"), CLOSED("closed rule"), WORKING("rule in work"); private String name; Status(String name) { this.name = name; } public String getName() { return this.name; } public String getId() { return this.name(); } }; }
Rule.Status.CLOSED
przetłumaczone na{id: "CLOSED", name: "closed rule"}
.źródło
Łatwym sposobem serializacji Enum jest użycie adnotacji @JsonFormat. @JsonFormat może skonfigurować serializację Enum na trzy sposoby.
@JsonFormat.Shape.STRING public Enum OrderType {...}
używa OrderType :: name jako metody serializacji. Serializacja OrderType.TypeA to
“TYPEA”
@JsonFormat.Shape.NUMBER Public Enum OrderTYpe{...}
używa OrderType :: ordinal jako metody serializacji. Serializacja OrderType.TypeA to
1
@JsonFormat.Shape.OBJECT Public Enum OrderType{...}
traktuje OrderType jako POJO. Serializacja OrderType.TypeA to
{"id":1,"name":"Type A"}
JsonFormat.Shape.OBJECT jest tym, czego potrzebujesz w swoim przypadku.
Twoje rozwiązanie jest nieco bardziej skomplikowane, określając serializator dla Enum.
Sprawdź to odniesienie: https://fasterxml.github.io/jackson-annotations/javadoc/2.2.0/com/fasterxml/jackson/annotation/JsonFormat.html
źródło
Użyj adnotacji @JsonCreator, utwórz metodę getType (), jest serializowany z toString lub działającym obiektem
{"ATIVO"}
lub
{"type": "ATIVO", "descricao": "Ativo"}
...
import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonFormat; import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.node.JsonNodeType; @JsonFormat(shape = JsonFormat.Shape.OBJECT) public enum SituacaoUsuario { ATIVO("Ativo"), PENDENTE_VALIDACAO("Pendente de Validação"), INATIVO("Inativo"), BLOQUEADO("Bloqueado"), /** * Usuarios cadastrados pelos clientes que não possuem acesso a aplicacao, * caso venham a se cadastrar este status deve ser alterado */ NAO_REGISTRADO("Não Registrado"); private SituacaoUsuario(String descricao) { this.descricao = descricao; } private String descricao; public String getDescricao() { return descricao; } // TODO - Adicionar metodos dinamicamente public String getType() { return this.toString(); } public String getPropertieKey() { StringBuilder sb = new StringBuilder("enum."); sb.append(this.getClass().getName()).append("."); sb.append(toString()); return sb.toString().toLowerCase(); } @JsonCreator public static SituacaoUsuario fromObject(JsonNode node) { String type = null; if (node.getNodeType().equals(JsonNodeType.STRING)) { type = node.asText(); } else { if (!node.has("type")) { throw new IllegalArgumentException(); } type = node.get("type").asText(); } return valueOf(type); } }
źródło
W Spring Boot 2 najłatwiej jest zadeklarować w pliku application.properties:
spring.jackson.serialization.WRITE_ENUMS_USING_TO_STRING=true spring.jackson.deserialization.READ_ENUMS_USING_TO_STRING=true
i zdefiniuj metodę toString () swoich wyliczeń.
źródło