【问题标题】:How do I deserialize Json with modifying the structure?如何通过修改结构反序列化 Json?
【发布时间】:2015-07-31 16:40:02
【问题描述】:

我有这个 Json 内容:

{
    "people":[
        {
            "name":"test1",
            "sirname":"test2",
            "details":{
                "social_no":1234567,
                "creadit_card_no":34582342309
            }
        },
        {
            "name":"test3",
            "sirname":"test4",
            "details":{
                "social_no":12345679,
                "creadit_card_no":345823423090
            }
        }
    ]
}

根据逻辑,这个 Json 应该有 3 个 POJO 类:一个将保存人员列表、人员对象和详细信息对象的类。

现在我的问题是,是否可以使用 Jackson 反序列化此 Json,或者如果无法使用 Jackson,则使用 GSON 库?一个包含人员列表,另一个(例如Human 类)具有以下结构:

public class Human{

    String name;
    String sirname;
    String social_no;
    String creadit_card_no;
    //..getters and setters
    //should correspond with this json fragment:
      // {
      //  "name":"test1",
      //  "sirname":"test2",
      //  "details":{
      //    "social_no":1234567,
      //    "creadit_card_no":34582342309
      // }
    }
}

如果可以的话,我该怎么做呢?

更新

我的实际 json 结构与这里给出的示例不同,所以here is the original json

所以我自己创建了一个TypeAdapter,下面是这个类的代码:

public class PlanTypeAdapter extends TypeAdapter<Plan> {
    private final String TAG = PlanTypeAdapter.class.getSimpleName();

    @Override
    public void write(JsonWriter out, Plan value) throws IOException {
        Log.d(TAG, "WRITE");
    }

    @Override
    public Plan read(JsonReader reader) throws IOException {
        Log.d(TAG, "READ");
        Plan plan = new Plan();
        if (reader.peek() == JsonToken.NULL) {
            reader.nextNull();
            return null;
        }

        reader.setLenient(false);
        while (reader.hasNext()) {
            Log.d(TAG, "PATH: " + reader.getPath());
            Log.d(TAG, "PEEK: " + reader.peek());
            if (reader.peek() == JsonToken.BEGIN_OBJECT) {
                Log.d(TAG, "BEGIN object, path: " + reader.getPath());
                reader.beginObject();
            } else if (reader.peek() == JsonToken.NULL) {
                Log.d(TAG, "NULL");
                reader.skipValue();
            } else if (reader.peek() == JsonToken.END_ARRAY) {
                Log.d(TAG, "END ARRAY");
                if (reader.getPath().contains("retailer")) {
                    reader.endObject();
                } else {
                    reader.endArray();
                }
            } else if (reader.peek() == JsonToken.END_OBJECT) {
                reader.endObject();
                Log.d(TAG, "END object, path: " + reader.getPath());
            } else if (reader.peek() == JsonToken.NUMBER) {
                Log.d(TAG, "NUMBER " + reader.getPath());
            } else if (reader.peek() == JsonToken.BOOLEAN) {
                Log.d(TAG, "BOOLEAN " + reader.getPath());
            } else if (reader.peek() == JsonToken.NAME) {
                switch (reader.nextName()) {
                    case "retailer":
                        reader.beginObject();
                        Log.d(TAG, "RET");
                        break;
                    case "national_plan":
                        reader.beginObject();
                        Log.d(TAG, "NPlan");
                        break;
                    case "name":
                        if (reader.getPath().contains("retailer")) {
                            plan.setRetailer_name(reader.nextString());
                            reader.skipValue();
                            reader.skipValue();
                            reader.endObject();
                        } else {
                            reader.skipValue();
                        }
                        break;
                    case "contract_end":
                        plan.setContract_end(reader.nextString());
                        break;
                    case "data_level_gb":
                        plan.setData_level_gb(reader.nextString());
                        break;
                    case "data_level_id":
                        plan.setData_level_id(reader.nextInt());
                        break;
                    case "days_to_end":
                        plan.setDays_to_switch(reader.nextInt());
                        break;
                    case "direct_from_operator":
                        plan.setDirect_from_operator(reader.nextBoolean());
                        break;
                    case "calculation_amount":
                        plan.setCalculationAmount(reader.nextDouble());
                        break;
                    case "network_generation_name":
                        plan.setNetwork_generation_(reader.nextString());
                        break;
                    case "partner_plan_id":
                        plan.setPartner_plan_id(reader.nextString());
                        break;
                    case "payment_level":
                        plan.setPayment_level(reader.nextString());
                        break;
                    case "payment_level_id":
                        plan.setPayment_level_id(reader.nextInt());
                        break;
                    case "roaming_amount":
                        plan.setRoaming_amount(reader.nextDouble());
                        break;
                    case "savings_amount":
                        plan.setSavings_amount(reader.nextDouble());
                        break;
                    case "savings_avg":
                        plan.setSavings_avg(reader.nextDouble());
                        break;
                    case "savings_percents":
                        plan.setSavings_percents(reader.nextInt());
                        break;
                    default:
                        Log.d(TAG, "DEFAULT " + reader.peek() + "");
                        reader.skipValue();
                        break;
                }

            } else {
                reader.skipValue();
            }
        }

        return plan;
    }
}

【问题讨论】:

  • 你为什么不用json2schema在线反序列化json

标签: java android json jackson gson


【解决方案1】:

如果你有一个非常非常大的文件,我建议使用Gson 的自定义反序列化器来执行此操作,但我不会使用JsonDeserializer 接口;使用TypeAdapter 接口,因为它性能更高(source)。我认为@codemonkey 有一个很好的答案,但它过于复杂,可以更简单地完成。具体来说,你永远不应该自己构建这些字符串(使用sb.append()),你应该远离JsonDeserializer

首先,创建您的自定义TypeAdapter

public class PersonTypeAdapter extends TypeAdapter<Person> {
  @Override
  public void write(JsonWriter out, Person value) throws IOException {
    if (value == null) {
      out.nullValue();
      return;
    }

    out.beginObject();
    out.name("name").value(value.name);
    out.name("sirname").value(value.sirname);
    out.name("details");
    out.beginObject();
    out.name("social_no").value(value.social_no);
    out.name("creadit_card_no").value(value.creadit_card_no);
    out.endObject();
    out.endObject();
  }

  @Override
  public Person read(JsonReader reader) throws IOException {
    if (reader.peek() == JsonToken.NULL) {
      reader.nextNull();
      return null;
    }

    reader.beginObject();
    validateName(reader, "name");
    String name = reader.nextString();
    validateName(reader, "sirname");
    String sirname = reader.nextString();
    validateName(reader, "details");
    reader.beginObject();
    validateName(reader, "social_no");
    String social_no = reader.nextString();
    validateName(reader, "creadit_card_no");
    String creadit_card_no = reader.nextString();
    reader.endObject();
    reader.endObject();
    return new Person(name, sirname, social_no, creadit_card_no);
  }

  private void validateName(JsonReader reader, String string) throws IOException {
    String name = reader.nextName();
    if(!string.equals(name)) {
      throw new JsonSyntaxException("Expected: \"" + string + "\", got \"" + name + "\"");
    }
  }
}

还有,你的 POJO,很明显:

public class Person {
  public final String name;
  public final String sirname;
  public final String social_no;
  public final String creadit_card_no;

  public Person(String name, String sirname, String social_no,
      String creadit_card_no) {
    this.name = name;
    this.sirname = sirname;
    this.social_no = social_no;
    this.creadit_card_no = creadit_card_no;
  }

  @Override
  public String toString() {
    return String.format(
        "Person [name=%s, sirname=%s, social_no=%s, creadit_card_no=%s]", name,
        sirname, social_no, creadit_card_no);
  }
}

然后,您可以使用此处的方法从文件中解析 Json。 /test.json 只是您在问题中给出的示例。

import java.io.InputStreamReader;
import java.io.Reader;
import java.util.List;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.annotations.SerializedName;

public class PersonExample {
  public static void main(String... args) {
    InputStreamReader streamReader = new InputStreamReader(
        PersonExample.class.getResourceAsStream("/test.json"));

    PeopleWrapper wrapper = parseJSON(streamReader);
    System.out.println(wrapper.people);
  }

  public static class PeopleWrapper {
    @SerializedName("people")
    public List<Person> people;
  }

  public static PeopleWrapper parseJSON(Reader jsonInput) {
    GsonBuilder builder = new GsonBuilder();
    builder.registerTypeAdapter(Person.class, new PersonTypeAdapter());
    Gson gson = builder.create();

    PeopleWrapper peopleWrapper = gson.fromJson(jsonInput, PeopleWrapper.class);

    return peopleWrapper;
  }
}

这个程序输出:

[Person [name=test1, sirname=test2, social_no=1234567, creadit_card_no=34582342309], Person [name=test3, sirname=test4, social_no=12345679, creadit_card_no=345823423090]]

所以你的实际问题比你最初描述的要复杂得多。我将向您展示您需要的TypeAdapter 的骨架,您可以找出其余的。基本上,按照您的做法创建 Plan 对象,然后为每个外部 JSON 键处理值。

  • 如果是一行,可以在switch语句中处理。
  • 如果是数组或对象,创建辅助方法来解析 JSON 的该部分。

您应该假设 JSON 格式正确,如果不是,让Gson 抛出异常。告诉它期待接下来会发生什么。

这里有一些代码向你展示这个想法:

import java.io.IOException;

import com.google.gson.*;
import com.google.gson.stream.*;

public class PlanTypeAdapter extends TypeAdapter<Plan> {
    private final String TAG = PlanTypeAdapter.class.getSimpleName();

    @Override
    public void write(JsonWriter out, Plan value) throws IOException {
        Log.d(TAG, "WRITE");
    }

    @Override
    public Plan read(JsonReader reader) throws IOException {
        Log.d(TAG, "READ");
        Plan plan = new Plan();
        if (reader.peek() == JsonToken.NULL) {
            reader.nextNull();
            return null;
        }

        reader.setLenient(false);
        reader.beginObject();

        while (!(reader.peek() == JsonToken.END_OBJECT)) {
            switch (reader.nextName()) {
            case "national_plan":
                handleNationalPlan(reader, plan);
                break;
            case "bill_total":
                handleBillTotal(reader, plan);
                break;
            case "contract_end":
                plan.setContract_end(reader.nextString());
                break;
            case "data_level_gb":
                plan.setData_level_gb(reader.nextString());
                break;
            case "data_level_id":
                plan.setData_level_id(reader.nextInt());
                break;
            case "days_to_end":
                plan.setDays_to_switch(reader.nextInt());
                break;
            case "direct_from_operator":
                plan.setDirect_from_operator(reader.nextBoolean());
                break;
            case "calculation_amount":
                plan.setCalculationAmount(reader.nextDouble());
                break;
            case "network_generation_name":
                plan.setNetwork_generation_(reader.nextString());
                break;
            case "partner_plan_id":
                plan.setPartner_plan_id(reader.nextString());
                break;
            case "payment_level":
                plan.setPayment_level(reader.nextString());
                break;
            case "payment_level_id":
                plan.setPayment_level_id(reader.nextInt());
                break;
            case "roaming_amount":
                plan.setRoaming_amount(reader.nextDouble());
                break;
            case "savings_amount":
                plan.setSavings_amount(reader.nextDouble());
                break;
            case "savings_avg":
                plan.setSavings_avg(reader.nextDouble());
                break;
            case "savings_percents":
                plan.setSavings_percents(reader.nextInt());
                break;
            case "yearly_id":
            case "handset":
            case "internals":
            case "consumer_id":
            case "calculation_details":
            case "operator":
            case "total":
            case "international_plan":
            case "contract_length":
            case "zone":
            case "externals":
            case "cancel_fee":
            case "transformers":
            case "one-offs":
            case "flow":
            case "roaming_plan":
            case "_id":
            // You can use this to ignore the keys you don't care about
            default:
                Log.d(TAG, "DEFAULT " + reader.peek() + "");
                reader.skipValue();
                break;
            }
        }

        reader.endObject();

        return plan;
    }

    private void handleNationalPlan(JsonReader reader, Plan plan) throws IOException {
        reader.beginObject();

        while (!(reader.peek() == JsonToken.END_OBJECT)) {
            switch(reader.nextName()) {
            case "contract_length":
                break;
            case "name":
                break;
            case "country":
            // etc.
            }
        }

        reader.endObject();
    }

    private void handleBillTotal(JsonReader reader, Plan plan) throws IOException {

    }

    // etc.
}

【讨论】:

  • 一个问题,我应该按照 json 结构顺序还是可以自定义获取值,例如:我可以得到 creadit_card_no,然后是 social_no
  • @DarkoPetkovski 通常 JSON 不保证排序,但这取决于特定文件的结构。如果您需要它来工作,无论顺序如何,我都可以修改答案。
  • 首先,我真的很感谢您的帮助,我认为这将是这个问题的答案。但是我在反序列化我拥有的 json 时仍然遇到一些问题。如果可以 - 请告诉我如何解析这个 json 文件。它与我发布的原始 json 示例不同,但它具有相似的结构。那么你可以检查这个文件并修改你的答案,以便我可以解析这个 json 文件吗? pastebin.com/3FWSs8Bf 我不需要所有的值,我只需要其中一个。如果你有空,我们可以聊聊吗?
  • 真的..?你能帮我解决这个问题吗?
【解决方案2】:

目前,Jackson 似乎不支持开箱即用的功能来映射嵌套路径中的字段。 有一个open issue 要求提供这样的功能,但这是一个何时完成的问题。 相反,可以使用@JsonUnwrapped 注释将嵌套对象序列化为 json 中的第一级属性。

因此,为了克服这个问题,似乎唯一的方法是编写一个自定义反序列化器,您可以将其映射到您的类,并根据需要使用它来创建类的实例。

【讨论】:

【解决方案3】:

使用 gson 库可以通过多种方式解析 json。我给你举两个例子。

方法 1 - 编写自定义反序列化器。该技术使用一个类来反序列化 person 对象。自定义反序列化器允许您使用 json 数据创建任何您想要的对象。以下是执行此操作所需的类:

Group.java:

public class Group {
    @SerializedName("people")
    private List<Person> persons;

    public List<Person> getPersons() {
        return persons;
    }

    public void setPersons(List<Person> persons) {
        this.persons = persons;
    }

    @Override
    public String toString() {
        String NEW_LINE = System.getProperty("line.separator");

        StringBuilder sb = new StringBuilder(this.getClass().getName());
        sb.append("{");
        sb.append(NEW_LINE);

        for(Person p : persons){
            sb.append(p.toString());
        }

        sb.append("}");
        return sb.toString();
    }
}

GsonTest.java:

public class GsonTest {

    public static void main(String[] args) {
        GsonBuilder gsonBuilder = new GsonBuilder();
        gsonBuilder.registerTypeAdapter(Person.class, new PersonDeserializer());
        Gson gson = gsonBuilder.create();

        try {
            JsonParser parser = new JsonParser();
            Object obj = parser.parse(new FileReader("C://data.json"));
            JsonObject jsonObject = (JsonObject) obj;

            Group group = gson.fromJson(jsonObject, Group.class);
            System.out.println(group.toString());
        } catch (JsonIOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (JsonSyntaxException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (FileNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

Person.java:

public class Person {

    public Person(String name, String sirname, Long social_no, Long creadit_card_no) {
        this.name = name;
        this.sirname = sirname;
        this.social_no = social_no;
        this.creadit_card_no = creadit_card_no;
    }

    private String name;
    private String sirname;
    private Long social_no;
    private Long creadit_card_no;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSirname() {
        return sirname;
    }

    public void setSirname(String sirname) {
        this.sirname = sirname;
    }

    public Long getSocial_no() {
        return social_no;
    }

    public void setSocial_no(Long social_no) {
        this.social_no = social_no;
    }

    public Long getCreadit_card_no() {
        return creadit_card_no;
    }

    public void Long(Long creadit_card_no) {
        this.creadit_card_no = creadit_card_no;
    }

    @Override
    public String toString() {
        String NEW_LINE = System.getProperty("line.separator");

        StringBuilder sb = new StringBuilder(this.getClass().getName());
        sb.append("{");
        sb.append(NEW_LINE);
        sb.append("name: ");
        sb.append(name);
        sb.append(NEW_LINE);

        sb.append("sirname: ");
        sb.append(sirname);
        sb.append(NEW_LINE);

        sb.append("social_no: ");
        sb.append(social_no);
        sb.append(NEW_LINE);

        sb.append("creadit_card_no: ");
        sb.append(creadit_card_no);
        sb.append(NEW_LINE);

        sb.append("}");
        sb.append(NEW_LINE);

        return sb.toString();
    }
}

PersonDeserializer.java

public class PersonDeserializer implements JsonDeserializer<Person> {

    public Person deserialize(JsonElement json, Type typeOfT,
        JsonDeserializationContext context) throws JsonParseException {

        JsonObject jsonObject = json.getAsJsonObject();

        String name = jsonObject.get("name").getAsString();
        String sirname = jsonObject.get("sirname").getAsString();

        JsonObject details = jsonObject.get("details").getAsJsonObject();

        Long social_no = details.get("social_no").getAsLong();
        Long creadit_card_no = details.get("creadit_card_no").getAsLong();

        Person person = new Person(name, sirname, social_no, creadit_card_no );

        return person;
    }
}

方法二 - 使用 JsonReader 类解析 json 数据。您不必使用此技术一次加载整个 json 文件。这是在资源有限的设备上解析大量数据的更好方法。但是,如果 json 结构发生变化,此代码将更难维护。我的示例代码受到这篇文章 http://developer.android.com/reference/android/util/JsonReader.html 的启发。将上面的 Person 类与这个新的 GsonTest 类一起使用:

public class GsonTest {
    List<Person> people = null;

    public GsonTest() {
        people = new ArrayList<Person>();
    }

    public static void main(String[] args) {
        GsonTest gt = new GsonTest();

        gt.doGson();
    }

    void doGson() {
        try {
            InputStream is = GsonTest.class.getResourceAsStream("data.json");

            JsonReader jsonReader = new JsonReader(new InputStreamReader(is, "UTF-8"));

            jsonReader.beginObject();

            while (jsonReader.hasNext()) {
                String name = jsonReader.nextName();
                if (name.equals("people")) {
                    readPeopleArray(jsonReader);
                }
            }

            jsonReader.endObject();

            for(Person p : people){
                System.out.println(p.toString());
            }
        }
        catch (NullPointerException e){
            e.printStackTrace();
        }
        catch (UnsupportedEncodingException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

    private void readPeopleArray(JsonReader jsonReader) throws IOException {
        jsonReader.beginArray();
        while (jsonReader.hasNext()) {
            readPersonObject(jsonReader);
        }
        jsonReader.endArray();
    }

    private void readPersonObject(JsonReader jsonReader) throws IOException {
        String name = null;
        String sirname = null;
        Long social_no = null;
        Long creadit_card_no = null;

        jsonReader.beginObject();
        while(jsonReader.hasNext()){
            String key = jsonReader.nextName();

            if(key.equals("details")){

                jsonReader.beginObject();

                while(jsonReader.hasNext()){
                    String detailKey = jsonReader.nextName();

                    if(detailKey.equals("social_no")){
                        social_no = jsonReader.nextLong();
                    }
                    else if(detailKey.equals("creadit_card_no")){
                        creadit_card_no = jsonReader.nextLong();
                    }
                    else{
                        jsonReader.skipValue();
                    }
                }

                jsonReader.endObject();
            }
            else if(key.equals("name")){
                name = jsonReader.nextString();
            }
            else if(key.equals("sirname")){
                sirname = jsonReader.nextString();
            }
        }
        jsonReader.endObject();

        people.add(new Person(name, sirname, social_no, creadit_card_no));
    }
}

【讨论】:

  • 感谢您的回答,但我可以这样做,这里的问题是我需要创建问题中描述的对象 - 不是纯 json 解析
  • 对不起,我误读了您的问题。查看我对 Person 类所做的更改。我使用 setDetails 方法来设置 social_no 和 creadit_card_no 的值。我相信这会创建您需要的对象。
  • 啊,你又在创建 Details 对象了。问题是我处理的真正的 json 文件非常大,所以我无法创建其中的所有对象。我需要一种方法来从 json 中创建自定义对象,只需要我需要的数据。
  • 你可以使用不同的 json 解析器吗?我建议尝试 Gson。您可以编写一个反序列化程序,将您需要的数据映射到您的 Person 类,而无需创建 Details 对象。这里有一篇文章供参考:javacreed.com/gson-deserialiser-example
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2015-03-19
  • 1970-01-01
  • 1970-01-01
  • 2016-09-17
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多