ใช้ Enums ขณะแยกวิเคราะห์ JSON กับ GSON


119

คำถามนี้เกี่ยวข้องกับคำถามก่อนหน้านี้ที่ฉันถามก่อนหน้านี้

JSON แยกวิเคราะห์โดยใช้ Gson

ฉันพยายามแยกวิเคราะห์ JSON เดียวกัน แต่ตอนนี้ฉันได้เปลี่ยนชั้นเรียนเล็กน้อยแล้ว

{
    "lower": 20,
    "upper": 40,
    "delimiter": " ",
    "scope": ["${title}"]
}

ชั้นเรียนของฉันดูเหมือนว่า:

public class TruncateElement {

   private int lower;
   private int upper;
   private String delimiter;
   private List<AttributeScope> scope;

   // getters and setters
}


public enum AttributeScope {

    TITLE("${title}"),
    DESCRIPTION("${description}"),

    private String scope;

    AttributeScope(String scope) {
        this.scope = scope;
    }

    public String getScope() {
        return this.scope;
    }
}

รหัสนี้แสดงข้อยกเว้น

com.google.gson.JsonParseException: The JsonDeserializer EnumTypeAdapter failed to deserialized json object "${title}" given the type class com.amazon.seo.attribute.template.parse.data.AttributeScope
at 

ข้อยกเว้นเป็นสิ่งที่เข้าใจได้เนื่องจากตามการแก้ปัญหาสำหรับคำถามก่อนหน้าของฉัน GSON คาดหวังว่าวัตถุ Enum จะถูกสร้างขึ้นเป็น

${title}("${title}"),
${description}("${description}");

แต่เนื่องจากสิ่งนี้เป็นไปไม่ได้ในเชิงไวยากรณ์คำแนะนำวิธีแก้ปัญหาที่แนะนำคืออะไร?

คำตอบ:


57

จากเอกสารสำหรับ Gson :

Gson จัดเตรียมการทำให้เป็นอนุกรมเริ่มต้นและการดีซีเรียลไลเซชันสำหรับ Enums ... หากคุณต้องการเปลี่ยนการแทนค่าเริ่มต้นคุณสามารถทำได้โดยการลงทะเบียนอะแด็ปเตอร์ประเภทผ่าน GsonBuilder.registerTypeAdapter (Type, Object)

ต่อไปนี้เป็นแนวทางหนึ่ง

import java.io.FileReader;
import java.lang.reflect.Type;
import java.util.List;

import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.JsonDeserializationContext;
import com.google.gson.JsonDeserializer;
import com.google.gson.JsonElement;
import com.google.gson.JsonParseException;

public class GsonFoo
{
  public static void main(String[] args) throws Exception
  {
    GsonBuilder gsonBuilder = new GsonBuilder();
    gsonBuilder.registerTypeAdapter(AttributeScope.class, new AttributeScopeDeserializer());
    Gson gson = gsonBuilder.create();

    TruncateElement element = gson.fromJson(new FileReader("input.json"), TruncateElement.class);

    System.out.println(element.lower);
    System.out.println(element.upper);
    System.out.println(element.delimiter);
    System.out.println(element.scope.get(0));
  }
}

class AttributeScopeDeserializer implements JsonDeserializer<AttributeScope>
{
  @Override
  public AttributeScope deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context)
      throws JsonParseException
  {
    AttributeScope[] scopes = AttributeScope.values();
    for (AttributeScope scope : scopes)
    {
      if (scope.scope.equals(json.getAsString()))
        return scope;
    }
    return null;
  }
}

class TruncateElement
{
  int lower;
  int upper;
  String delimiter;
  List<AttributeScope> scope;
}

enum AttributeScope
{
  TITLE("${title}"), DESCRIPTION("${description}");

  String scope;

  AttributeScope(String scope)
  {
    this.scope = scope;
  }
}

311

ฉันต้องการขยายคำตอบ NAZIK / user2724653 เล็กน้อย (สำหรับกรณีของฉัน) นี่คือรหัส Java:

public class Item {
    @SerializedName("status")
    private Status currentState = null;

    // other fields, getters, setters, constructor and other code...

    public enum Status {
        @SerializedName("0")
        BUY,
        @SerializedName("1")
        DOWNLOAD,
        @SerializedName("2")
        DOWNLOADING,
        @SerializedName("3")
        OPEN
     }
}

ในไฟล์ json คุณมีเพียงฟิลด์"status": "N",โดยที่ N = 0,1,2,3 - ขึ้นอยู่กับค่าสถานะ นั่นคือทั้งหมดที่GSONทำงานได้ดีกับค่าของenumคลาสที่ซ้อนกัน ในกรณีของฉันฉันได้แยกวิเคราะห์รายการItemsจากjsonอาร์เรย์:

List<Item> items = new Gson().<List<Item>>fromJson(json,
                                          new TypeToken<List<Item>>(){}.getType());

28
คำตอบนี้แก้ปัญหาทุกอย่างได้อย่างสมบูรณ์แบบไม่ต้องใช้อะแดปเตอร์ประเภท
Lena Bru

4
เมื่อฉันทำสิ่งนี้ด้วย Retrofit / Gson SerializedName ของค่า enum จะเพิ่มเครื่องหมายคำพูดพิเศษ เซิร์ฟเวอร์ได้รับจริง"1"เช่นแทนที่จะเป็นเพียง1...
Matthew Housser

17
จะเกิดอะไรขึ้นถ้า json ที่มีสถานะ 5 จะมาถึง? มีวิธีใดในการกำหนดค่าเริ่มต้นหรือไม่?
DmitryBorodin

8
@DmitryBorodin ถ้าค่าจาก JSON ไม่ตรงกับใด ๆSerializedNameแล้ว enum nullจะเริ่มต้นที่จะ พฤติกรรมเริ่มต้นของสถานะที่ไม่รู้จักสามารถจัดการได้ในคลาส Wrapper อย่างไรก็ตามหากคุณต้องการตัวแทนสำหรับ "ไม่ทราบ" นอกเหนือจากnullจากนั้นคุณจะต้องเขียน deserializer แบบกำหนดเองหรืออะแดปเตอร์ชนิด
Peter F

32

ใช้คำอธิบายประกอบ@SerializedName:

@SerializedName("${title}")
TITLE,
@SerializedName("${description}")
DESCRIPTION

9

ด้วย GSON เวอร์ชัน 2.2.2 enum จะได้รับการมาร์แชลและ unmarshalled ได้อย่างง่ายดาย

import com.google.gson.annotations.SerializedName;

enum AttributeScope
{
  @SerializedName("${title}")
  TITLE("${title}"),

  @SerializedName("${description}")
  DESCRIPTION("${description}");

  private String scope;

  AttributeScope(String scope)
  {
    this.scope = scope;
  }

  public String getScope() {
    return scope;
  }
}

8

ตัวอย่างต่อไปนี้จะขจัดความต้องการที่ชัดเจนGson.registerTypeAdapter(...)โดยใช้@JsonAdapter(class)คำอธิบายประกอบซึ่งมีให้ตั้งแต่ Gson 2.3 (ดูความคิดเห็นpm_labs )

@JsonAdapter(Level.Serializer.class)
public enum Level {
    WTF(0),
    ERROR(1),
    WARNING(2),
    INFO(3),
    DEBUG(4),
    VERBOSE(5);

    int levelCode;

    Level(int levelCode) {
        this.levelCode = levelCode;
    }

    static Level getLevelByCode(int levelCode) {
        for (Level level : values())
            if (level.levelCode == levelCode) return level;
        return INFO;
    }

    static class Serializer implements JsonSerializer<Level>, JsonDeserializer<Level> {
        @Override
        public JsonElement serialize(Level src, Type typeOfSrc, JsonSerializationContext context) {
            return context.serialize(src.levelCode);
        }

        @Override
        public Level deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) {
            try {
                return getLevelByCode(json.getAsNumber().intValue());
            } catch (JsonParseException e) {
                return INFO;
            }
        }
    }
}

1
โปรดทราบว่าหมายเหตุนี้ใช้ได้เฉพาะรุ่นเริ่มต้นที่ 2.3: google.github.io/gson/apidocs/index.html?com/google/gson/...
pm_labs

3
ระมัดระวังในการเพิ่มคลาส serializer / deserializer ของคุณในการกำหนดค่า proguard ของคุณเนื่องจากอาจถูกลบออก (มันเกิดขึ้นกับฉัน)
TormundThunderfist

2

หากคุณต้องการใช้ค่าลำดับของ Enum คุณสามารถลงทะเบียนโรงงานอะแด็ปเตอร์ชนิดเพื่อแทนที่โรงงานเริ่มต้นของ Gson

public class EnumTypeAdapter <T extends Enum<T>> extends TypeAdapter<T> {
    private final Map<Integer, T> nameToConstant = new HashMap<>();
    private final Map<T, Integer> constantToName = new HashMap<>();

    public EnumTypeAdapter(Class<T> classOfT) {
        for (T constant : classOfT.getEnumConstants()) {
            Integer name = constant.ordinal();
            nameToConstant.put(name, constant);
            constantToName.put(constant, name);
        }
    }
    @Override public T read(JsonReader in) throws IOException {
        if (in.peek() == JsonToken.NULL) {
            in.nextNull();
            return null;
        }
        return nameToConstant.get(in.nextInt());
    }

    @Override public void write(JsonWriter out, T value) throws IOException {
        out.value(value == null ? null : constantToName.get(value));
    }

    public static final TypeAdapterFactory ENUM_FACTORY = new TypeAdapterFactory() {
        @SuppressWarnings({"rawtypes", "unchecked"})
        @Override public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> typeToken) {
            Class<? super T> rawType = typeToken.getRawType();
            if (!Enum.class.isAssignableFrom(rawType) || rawType == Enum.class) {
                return null;
            }
            if (!rawType.isEnum()) {
                rawType = rawType.getSuperclass(); // handle anonymous subclasses
            }
            return (TypeAdapter<T>) new EnumTypeAdapter(rawType);
        }
    };
}

จากนั้นเพียงแค่ลงทะเบียนโรงงาน

Gson gson = new GsonBuilder()
               .registerTypeAdapterFactory(EnumTypeAdapter.ENUM_FACTORY)
               .create();

0

ใช้วิธีนี้

GsonBuilder.enableComplexMapKeySerialization();

3
แม้ว่ารหัสนี้อาจตอบคำถามได้ แต่การให้บริบทเพิ่มเติมเกี่ยวกับวิธีการและ / หรือเหตุผลในการแก้ปัญหาจะช่วยเพิ่มมูลค่าในระยะยาวของคำตอบ
Nic3500

ตาม gson 2.8.5 สิ่งนี้จำเป็นเพื่อใช้คำอธิบายประกอบ SerializedName บน enums ที่คุณต้องการใช้เป็นคีย์
vazor
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.