Serialisierung von Enums mit Jackson

90

Ich habe eine Enum unten beschrieben:

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....
}

Ich gebe mit meinem Controller ( new OrderType[] {UNKNOWN,TYPEA,TYPEB,TYPEC};) ein Enum-Array zurück und Spring serialisiert es in den folgenden JSON-String:

["UNKNOWN", "TYPEA", "TYPEB", "TYPEC"] 

Was ist der beste Ansatz, um Jackson zu zwingen, Enums genau wie POJOs zu serialisieren? Z.B:

[
  {"id": 1, "name": "Undefined"},
  {"id": 2, "name": "Type A"},
  {"id": 3, "name": "Type B"},
  {"id": 4, "name": "Type C"}
]

Ich habe mit verschiedenen Anmerkungen gespielt, konnte aber kein solches Ergebnis erzielen.

Kein Schicksal
quelle
1
Sieht so aus, als hätten Sie die Lösung bereits gefunden. großartig! War neugierig, warum du es brauchst?
StaxMan
Ich entwickle eine GWT-Anwendung, die über JSON serverseitig kommuniziert. Diese Aufzählung enthält Optionswerte für die Combobox.
Nofate
Ah ok. Also eine Art Kurzschrift für eine Reihe von Werten ... interessant.
StaxMan

Antworten:

87

Endlich habe ich selbst eine Lösung gefunden.

Ich musste enum mit dem @JsonSerialize(using = OrderTypeSerializer.class)benutzerdefinierten Serializer kommentieren und implementieren:

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();
  }
}
Kein Schicksal
quelle
4
Beachten Sie, dass zum Konfigurieren von Jackson für die Verwendung der benutzerdefinierten (De-) Serialisierungsverarbeitung eine Alternative zur Verwendung einer Anmerkung darin besteht, (De-) Serialisierer bei einem Konfigurationsmodul zu registrieren. wiki.fasterxml.com/JacksonHowToCustomSerializers
Programmierer Bruce
1
Dies hat bei mir mit Spring 3.1.1 nicht funktioniert. Mein @Controller gibt immer noch json ohne meine Attribute zurück.
Dave
Ich habe einige Aufzählungen und möchte alle Aufzählungen mit einer Funktion erhalten. Wie kann ich es tun?
Morteza Malvandi
Für einen Aufzählungstyp muss ich einen benutzerdefinierten Deserializer definieren. Gibt es eine generische Lösung?
Chao
78
@JsonFormat(shape= JsonFormat.Shape.OBJECT)
public enum SomeEnum

verfügbar seit https://github.com/FasterXML/jackson-databind/issues/24

gerade getestet funktioniert es mit Version 2.1.2

Antwort auf TheZuck :

Ich habe dein Beispiel ausprobiert und Json verstanden:

{"events":[{"type":"ADMIN"}]}

Mein Code:

@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;
 }
}

und Abhängigkeiten sind:

<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>
Vecnas
quelle
2
Ich mag diese Alternative, sie ist sauberer, aber ich habe sie mit dieser Klasse ausprobiert und der Typ wird nicht serialisiert. Irgendeine Idee, was falsch ist? @JsonFormat (shape = JsonFormat.Shape.OBJECT) @JsonAutoDetect () public enum Event {VISIT_WEBSITE (Type.ADMIN); @JsonProperty public Type type; public Type getType () {Rückgabetyp; } Ereignis (Typ Typ) {this.type = Typ; } öffentliche Aufzählung Typ {ADMIN, CONSUMER,}} Ich verwende Jackson 2.1.2
TheZuck
Ich habe zusätzliche Details zu Antwortkörper hinzugefügt
Vecnas
Als ich herausfand, was falsch war, verwendete ich Jackson 2.1.2, aber meine Spring-Version war immer noch 3.1, daher wurde diese Version nicht unterstützt. Auf 3.2.1 aktualisiert und jetzt ist alles in Ordnung. Vielen Dank!
TheZuck
@Vecnas Kann ich die Standardeinstellung @JsonFormatder Aufzählung überschreiben, wenn sie in einer anderen Entität verwendet wird? Zum Beispiel eine Entität, in der ich möchte, dass die Aufzählung als Zeichenfolge anstelle eines Objekts serialisiert wird. Ich versuche, ein weiteres @JsonFormatFeld in der Klasse hinzuzufügen, das die Aufzählung verwendet, aber es wird immer als Objekt serialisiert.
Herau
Was ich gefunden habe, benutze - @JsonSerialize (using = ToStringSerializer.class) für ein Feld, es verwendet toString (). Keine strenge Lösung, funktioniert aber
Vecnas
25

Ich habe eine sehr schöne und prägnante Lösung gefunden, die besonders nützlich ist, wenn Sie die Enum-Klassen nicht wie in meinem Fall ändern können. Anschließend sollten Sie einen benutzerdefinierten ObjectMapper mit einer bestimmten aktivierten Funktion bereitstellen. Diese Funktionen sind seit Jackson 1.6 verfügbar.

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);
    }
}

Weitere enumbezogene Funktionen sind verfügbar, siehe hier:

https://github.com/FasterXML/jackson-databind/wiki/Serialization-features https://github.com/FasterXML/jackson-databind/wiki/Deserialization-Features

Lagivan
quelle
4
Genau. Darüber hinaus benötigen Sie in Jackson 2.5 keinen benutzerdefinierten Objekt-Mapper. Tun Sie dies einfach: objMapper.enable(SerializationFeature.WRITE_ENUMS_USING_TO_STRING);und dies:objMapper.enable(DeserializationFeature.READ_ENUMS_USING_TO_STRING);
Jake Toronto
14

Hier ist meine Lösung. Ich möchte, dass sich die Aufzählung verwandelt {id: ..., name: ...}.

Mit 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);
    }
}

Mit 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übersetzt in {id: "CLOSED", name: "closed rule"}.

Gavenkoa
quelle
Ausgezeichnet. Du hast meinen Tag gerettet :-)
Sriram
4

Eine einfache Möglichkeit, Enum zu serialisieren, ist die Verwendung der Annotation @JsonFormat. @JsonFormat kann die Serialisierung einer Aufzählung auf drei Arten konfigurieren.

@JsonFormat.Shape.STRING
public Enum OrderType {...}

verwendet OrderType :: name als Serialisierungsmethode. Serialisierung von OrderType.TypeA ist“TYPEA”

@JsonFormat.Shape.NUMBER
Public Enum OrderTYpe{...}

verwendet OrderType :: ordinal als Serialisierungsmethode. Serialisierung von OrderType.TypeA ist1

@JsonFormat.Shape.OBJECT
Public Enum OrderType{...}

behandelt OrderType als POJO. Serialisierung von OrderType.TypeA ist{"id":1,"name":"Type A"}

JsonFormat.Shape.OBJECT ist das, was Sie in Ihrem Fall benötigen.

Etwas komplizierter ist Ihre Lösung, indem Sie einen Serializer für die Enum angeben.

Überprüfen Sie diese Referenz: https://fasterxml.github.io/jackson-annotations/javadoc/2.2.0/com/fasterxml/jackson/annotation/JsonFormat.html

Strahl
quelle
3

Verwenden Sie die Annotation @JsonCreator, erstellen Sie die Methode getType (), serialisieren Sie mit toString oder arbeiten Sie mit Objekten

{"ATIVO"}

oder

{"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);
    }

}
Gleidosn
quelle
0

In Spring Boot 2 ist es am einfachsten, in Ihrer application.properties zu deklarieren:

spring.jackson.serialization.WRITE_ENUMS_USING_TO_STRING=true
spring.jackson.deserialization.READ_ENUMS_USING_TO_STRING=true

und definieren Sie die toString () -Methode Ihrer Aufzählungen.

JRA_TLL
quelle