Jackson ile numaralandırılıyor


90

Aşağıda açıklanan bir Enum var:

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

Controller ( new OrderType[] {UNKNOWN,TYPEA,TYPEB,TYPEC};) ile enum dizisini döndürüyorum ve Spring onu aşağıdaki json dizesine serileştiriyor:

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

Jackson'ı POJO'lar gibi numaralandırmaları serileştirmeye zorlamak için en iyi yaklaşım nedir? Örneğin:

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

Farklı notlarla oynadım ama böyle bir sonuç alamadım.


1
Görünüşe göre çözümü zaten buldunuz; harika! Neden ihtiyacın olduğunu merak ediyor muydun?
StaxMan

JSON aracılığıyla sunucu tarafı ile iletişim kuran bir GWT uygulaması geliştiriyorum. Bu numaralandırma, birleşik giriş kutusu için seçenek değerleri sağlayacaktır.
Nofate

Ah tamam. Bu yüzden bir dizi değer için kısa el ... ilginç.
StaxMan

Yanıtlar:


87

Sonunda çözümü kendim buldum.

Numaralandırmaya ek açıklama eklemek @JsonSerialize(using = OrderTypeSerializer.class)ve özel serileştirici uygulamak zorunda kaldım :

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

4
Jackson'ı özel (de) serileştirme işlemini kullanacak şekilde yapılandırmak için, bir açıklama kullanmaya alternatif olarak serileştiricileri bir yapılandırma modülü ile kaydetmek (de) yapmaktır. wiki.fasterxml.com/JacksonHowToCustomSerializers
Programcı Bruce

1
Bu, Bahar 3.1.1'i kullanırken benim için işe yaramadı. @Controller, özelliklerim olmadan hala json döndürüyor.
Dave

Bazı Numaralandırmalarım var ve Tüm numaralandırmaları tek bir işlevle almak istiyorum. Nasıl yapabilirim?
Morteza Malvandi

Bir enum türü için, özel bir seriyi kaldırıcı tanımlamam gerekiyor. Herhangi bir genel çözüm var mı?
Chao

78
@JsonFormat(shape= JsonFormat.Shape.OBJECT)
public enum SomeEnum

https://github.com/FasterXML/jackson-databind/issues/24 tarihinden itibaren mevcuttur

2.1.2 sürümüyle çalıştığını test etti

TheZuck'a cevap :

Örneğinizi denedim, Json aldım:

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

Kodum:

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

ve bağımlılıklar şunlardır:

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

2
Bu alternatifi beğendim, daha temiz, ancak bu sınıfla denedim ve tür serileştirilmiyor, sorun ne? @JsonFormat (şekil = JsonFormat.Shape.OBJECT) @JsonAutoDetect () genel enum Olay {VISIT_WEBSITE (Type.ADMIN); @JsonProperty genel Tür türü; public Type getType () {dönüş türü; } Olay (Tür türü) {this.type = tür; } public enum Type {ADMIN, CONSUMER,}} Jackson 2.1.2 kullanıyorum
TheZuck

Ben cevap gövdesine ek ayrıntılar ekledik
Vecnas

neyin yanlış olduğunu öğrendim, Jackson 2.1.2 kullanıyordum ama Spring versiyonum hala 3.1 idi, dolayısıyla bu versiyonu desteklemiyordu. 3.2.1'e yükseltildi ve şimdi her şey yolunda. Teşekkürler!
TheZuck

@Vecnas @JsonFormatBaşka bir varlıkta kullanıldığında numaralamanın varsayılanını geçersiz kılabilir miyim ? örneğin, numaralandırmanın bir nesne yerine bir dizge olarak serileştirilmesini istediğim bir varlık. @JsonFormatSınıftaki alana numaralandırmayı kullanan ancak her zaman bir nesne olarak serileştirilen başka bir alan eklemeye çalışıyorum .
herau

Bulduğum şey, bir alan için - @JsonSerialize kullanın (= ToStringSerializer.class kullanarak), toString () kullanıyor. Kesin bir çözüm değil, ama işe yarıyor
Vecnas

25

Çok güzel ve öz bir çözüm buldum, özellikle benim durumumda olduğu gibi enum sınıflarını değiştiremediğinizde yararlıdır. Ardından, belirli bir özelliği etkinleştirilmiş özel bir ObjectMapper sağlamalısınız. Bu özellikler Jackson 1.6'dan beri mevcuttur.

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

Numaralandırmayla ilgili daha fazla özellik mevcuttur, buraya bakın:

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


4
Katılıyorum. Dahası, Jackson 2.5'te özel bir nesne eşleştiricisine ihtiyacınız yoktur. Sadece şunu yapın: objMapper.enable(SerializationFeature.WRITE_ENUMS_USING_TO_STRING);ve bu:objMapper.enable(DeserializationFeature.READ_ENUMS_USING_TO_STRING);
Jake Toronto

14

İşte benim çözümüm. Enum'u {id: ..., name: ...}biçimlendirmek istiyorum .

Jackson 1.x ile :

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 ile :

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.CLOSEDtercüme edildi {id: "CLOSED", name: "closed rule"}.


Mükemmel.
Günümü

4

Enum'u serileştirmenin kolay bir yolu, @JsonFormat açıklamasını kullanmaktır. @JsonFormat, Enum'un serileştirmesini üç şekilde yapılandırabilir.

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

Serileştirme yöntemi olarak OrderType :: name kullanır. OrderType.TypeA serileştirmesi“TYPEA”

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

serileştirme yöntemi olarak OrderType :: ordinal kullanır. OrderType.TypeA serileştirmesi1

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

OrderType'ı POJO olarak değerlendirir. OrderType.TypeA serileştirmesi{"id":1,"name":"Type A"}

JsonFormat.Shape.OBJECT , sizin durumunuzda ihtiyacınız olan şeydir.

Çözümünüz için biraz daha karmaşık bir yol, Enum için bir serileştirici belirlemektir.

Bu referansa göz atın: https://fasterxml.github.io/jackson-annotations/javadoc/2.2.0/com/fasterxml/jackson/annotation/JsonFormat.html


3

@JsonCreator ek açıklamasını kullanın, getType () yöntemini oluşturun, toString ile serileştirin veya nesne çalışıyor

{"ATIVO"}

veya

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

}

0

Spring Boot 2'de en kolay yol, application.properties dosyasında belirtmektir:

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

ve numaralandırmalarınızın toString () yöntemini tanımlayın.

Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.