У меня есть Enum, описанный ниже:
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....
}
Я возвращаю массив enum с помощью своего контроллера ( new OrderType[] {UNKNOWN,TYPEA,TYPEB,TYPEC};
), и Spring сериализует его в следующую строку json:
["UNKNOWN", "TYPEA", "TYPEB", "TYPEC"]
Как лучше всего заставить Джексона сериализовать перечисления так же, как POJO? Например:
[
{"id": 1, "name": "Undefined"},
{"id": 2, "name": "Type A"},
{"id": 3, "name": "Type B"},
{"id": 4, "name": "Type C"}
]
Я поигрался с разными аннотациями, но добиться такого результата не удалось.
Ответы:
Наконец я сам нашел решение.
Мне пришлось аннотировать enum
@JsonSerialize(using = OrderTypeSerializer.class)
и реализовать собственный сериализатор: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(); } }
источник
@JsonFormat(shape= JsonFormat.Shape.OBJECT) public enum SomeEnum
доступно с https://github.com/FasterXML/jackson-databind/issues/24
только что протестировал работает с версией 2.1.2
ответ TheZuck :
Я пробовал ваш пример, получил Json:
{"events":[{"type":"ADMIN"}]}
Мой код:
@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; } }
и зависимости:
<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>
источник
@JsonFormat
перечисления по умолчанию, если оно используется в другом объекте? например, сущность, в которой я хочу, чтобы перечисление было сериализовано как строка, а не как объект. Я пытаюсь добавить еще один@JsonFormat
в поле класса, который использует перечисление, но всегда сериализуется как объект.Я нашел очень красивое и краткое решение, особенно полезное, когда вы не можете изменять классы перечисления, как это было в моем случае. Затем вы должны предоставить настраиваемый ObjectMapper с включенной определенной функцией. Эти функции доступны с версии 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); } }
Доступны и другие функции, связанные с перечислением, см. Здесь:
https://github.com/FasterXML/jackson-databind/wiki/Serialization-features https://github.com/FasterXML/jackson-databind/wiki/Deserialization-Features
источник
objMapper.enable(SerializationFeature.WRITE_ENUMS_USING_TO_STRING);
и это:objMapper.enable(DeserializationFeature.READ_ENUMS_USING_TO_STRING);
Вот мое решение. Я хочу преобразовать enum в
{id: ..., name: ...}
форму.С Jackson 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); } }
С 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
переведено на{id: "CLOSED", name: "closed rule"}
.источник
Простой способ сериализации Enum - использовать аннотацию @JsonFormat. @JsonFormat может настроить сериализацию Enum тремя способами.
@JsonFormat.Shape.STRING public Enum OrderType {...}
использует OrderType :: name в качестве метода сериализации. Сериализация OrderType.TypeA - это
“TYPEA”
@JsonFormat.Shape.NUMBER Public Enum OrderTYpe{...}
использует OrderType :: ordinal в качестве метода сериализации. Сериализация OrderType.TypeA - это
1
@JsonFormat.Shape.OBJECT Public Enum OrderType{...}
обрабатывает OrderType как POJO. Сериализация OrderType.TypeA - это
{"id":1,"name":"Type A"}
JsonFormat.Shape.OBJECT - это то, что вам нужно в вашем случае.
Немного более сложный способ - это ваше решение, указав сериализатор для Enum.
Ознакомьтесь с этой ссылкой: https://fasterxml.github.io/jackson-annotations/javadoc/2.2.0/com/fasterxml/jackson/annotation/JsonFormat.html
источник
Используйте аннотацию @JsonCreator, создайте метод getType (), сериализуйте с toString или работающим объектом
{"ATIVO"}
или же
{"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); } }
источник
В Spring Boot 2 самый простой способ - объявить в application.properties:
spring.jackson.serialization.WRITE_ENUMS_USING_TO_STRING=true spring.jackson.deserialization.READ_ENUMS_USING_TO_STRING=true
и определите метод toString () ваших перечислений.
источник