【问题标题】:Converting JSON data to Java object将 JSON 数据转换为 Java 对象
【发布时间】:2010-12-13 20:55:13
【问题描述】:

我希望能够在我的 Java 操作方法中访问 JSON 字符串的属性。只需说出myJsonString = object.getJson() 即可获得该字符串。下面是字符串的示例:

{
    'title': 'ComputingandInformationsystems',
    'id': 1,
    'children': 'true',
    'groups': [{
        'title': 'LeveloneCIS',
        'id': 2,
        'children': 'true',
        'groups': [{
            'title': 'IntroToComputingandInternet',
            'id': 3,
            'children': 'false',
            'groups': []
        }]
    }]
}

在此字符串中,每个 JSON 对象都包含一组其他 JSON 对象。目的是提取 ID 列表,其中任何给定对象拥有包含其他 JSON 对象的组属性。我将 Google 的 Gson 视为一个潜在的 JSON 插件。任何人都可以就如何从这个 JSON 字符串生成 Java 提供某种形式的指导吗?

【问题讨论】:

标签: java json jackson gson


【解决方案1】:

我将 Google 的 Gson 视为一个潜在的 JSON 插件。任何人都可以就如何从这个 JSON 字符串生成 Java 提供某种形式的指导吗?

Google Gson 支持泛型和嵌套 bean。 JSON 中的 [] 表示一个数组,应该映射到一个 Java 集合,例如 List 或只是一个普通的 Java 数组。 JSON 中的 {} 表示一个对象,应该映射到 Java Map 或只是一些 JavaBean 类。

您有一个带有多个属性的 JSON 对象,其中的 groups 属性表示一组相同类型的嵌套对象。这可以通过以下方式用 Gson 解析:

package com.stackoverflow.q1688099;

import java.util.List;
import com.google.gson.Gson;

public class Test {

    public static void main(String... args) throws Exception {
        String json = 
            "{"
                + "'title': 'Computing and Information systems',"
                + "'id' : 1,"
                + "'children' : 'true',"
                + "'groups' : [{"
                    + "'title' : 'Level one CIS',"
                    + "'id' : 2,"
                    + "'children' : 'true',"
                    + "'groups' : [{"
                        + "'title' : 'Intro To Computing and Internet',"
                        + "'id' : 3,"
                        + "'children': 'false',"
                        + "'groups':[]"
                    + "}]" 
                + "}]"
            + "}";

        // Now do the magic.
        Data data = new Gson().fromJson(json, Data.class);

        // Show it.
        System.out.println(data);
    }

}

class Data {
    private String title;
    private Long id;
    private Boolean children;
    private List<Data> groups;

    public String getTitle() { return title; }
    public Long getId() { return id; }
    public Boolean getChildren() { return children; }
    public List<Data> getGroups() { return groups; }

    public void setTitle(String title) { this.title = title; }
    public void setId(Long id) { this.id = id; }
    public void setChildren(Boolean children) { this.children = children; }
    public void setGroups(List<Data> groups) { this.groups = groups; }
    
    public String toString() {
        return String.format("title:%s,id:%d,children:%s,groups:%s", title, id, children, groups);
    }
}

相当简单,不是吗?只要有一个合适的 JavaBean 并调用Gson#fromJson()

另见:

【讨论】:

  • 表演?你真的测量过吗?虽然 GSON 具有合理的功能集,但我认为性能是一种弱点(根据 [cowtowncoder.com/blog/archives/2009/09/entry_326.html])例如:我认为 GSON 并不真正需要设置器,而是基于字段。所以代码可以稍微简化一下。
  • 我在安卓应用中使用它。这不是最快的解决方案,但它足够简单,可以通过编程来证明到目前为止用户性能不足是合理的。也许在更高版本的应用程序中,它将被删除以获得更快的解决方案。
  • 写速度,够快就够快了。我只是评论了预期的良好表现。功能集明智的杰克逊处理所有相同的嵌套、分层、泛型,所以这不是速度差异的来源。拥有 getter 和 setter 不会以任何可衡量的方式影响性能(对于我知道的包),所以绝对可以拥有它们。
  • +1 表示“包 com.stackoverflow.q1688099;”。出于某种原因,这让我笑了。
  • 我使用过很多 java 和 json。没有什么能比得上杰克逊,StaxMan 是爸爸!!!!,GSon 是一个不错的尝试,但它没有得到雪茄:)
【解决方案2】:

如果您使用任何类型的特殊映射以及特殊映射的键或值,您会发现谷歌的实现没有考虑到它。

【讨论】:

    【解决方案3】:

    Gson 万岁!它非常酷,非常棒,但是当您想做除了简单对象之外的任何事情时,您可能很容易需要开始构建自己的序列化程序(这并不难)。

    此外,如果您有一个对象数组,并且您将一些 json 反序列化到该对象数组中,那么真正的类型就会丢失!完整的对象甚至不会被复制!使用 XStream.. 如果使用 jsondriver 并设置正确的设置,它将把丑陋的类型编码到实际的 json 中,这样你就不会丢失任何东西。真正的序列化需要付出很小的代价(丑陋的 json)。

    请注意,Jackson 修复了这些问题,并且是 faster 而不是 GSON。

    【讨论】:

    【解决方案4】:

    奇怪的是,到目前为止提到的唯一不错的 JSON 处理器是 GSON。

    这里有更多不错的选择:

    • Jackson (Github) -- 强大的数据绑定(JSON 到/从 POJO)、流式传输(超快)、树模型(方便无类型访问)
    • Flex-JSON -- 高度可配置的序列化

    编辑(2013 年 8 月):

    还有一点需要考虑:

    • Genson -- 类似于 Jackson 的功能,旨在让开发人员更容易配置

    【讨论】:

      【解决方案5】:

      如果通过任何更改,您所在的应用程序已经使用了http://restfb.com/,那么您可以这样做:

      import com.restfb.json.JsonObject;
      
      ...
      
      JsonObject json = new JsonObject(jsonString);
      json.get("title");
      

      等等

      【讨论】:

      • 您的解决方案更短且更易于理解,为什么它只获得 3 个赞?有什么问题吗?
      【解决方案6】:

      或与杰克逊:

      String json = "...";
      ObjectMapper m = new ObjectMapper();
      Set<Product> products = m.readValue(json, new TypeReference<Set<Product>>() {});
      

      【讨论】:

      • 这将给出错误 can not deserialize instance of java.util.HashSet out of START_OBJECT token
      【解决方案7】:
      HashMap keyArrayList = new HashMap();
      Iterator itr = yourJson.keys();
      while (itr.hasNext())
      {
          String key = (String) itr.next();
          keyArrayList.put(key, yourJson.get(key).toString());
      }
      

      【讨论】:

        【解决方案8】:

        标准的东西有什么问题?

        JSONObject jsonObject = new JSONObject(someJsonString);
        JSONArray jsonArray = jsonObject.getJSONArray("someJsonArray");
        String value = jsonArray.optJSONObject(i).getString("someJsonValue");
        

        【讨论】:

        • 它非常慢:github.com/fabienrenaud/java-json-benchmark 最近在工作中,我通过切换所有 org.json ser/deserialization 调用将我们的 prod 服务器的性能(cpu 使用率减少了一半,延迟减少了)翻了一番使用杰克逊。
        【解决方案9】:

        试一试:

        https://github.com/RichardHightower/boon

        这是邪恶的快:

        https://github.com/RichardHightower/json-parsers-benchmark

        不要相信我的话...查看 gatling 基准测试。

        https://github.com/gatling/json-parsers-benchmark

        (在某些情况下高达 4x,并且在 100 次测试中。它还有一个更快的索引覆盖模式。它很年轻但已经有一些用户。)

        它可以将 JSON 解析为 Maps 和 Lists,比任何其他 lib 解析为 JSON DOM 的速度都要快,并且没有索引覆盖模式。使用 Boon Index Overlay 模式,它甚至更快。

        它还有一个非常快速的 JSON 宽松模式和一个 PLIST 解析器模式。 :)(并且具有超低内存,直接从字节模式使用 UTF-8 编码)。

        它也有最快的 JSON to JavaBean 模式。

        它是新的,但如果您正在寻找速度和简单的 API,我认为没有更快或更简约的 API。

        【讨论】:

        • 您能否提供最新版本的最新文档链接?截至今天,我找到了 0.4,但无法轻松找到该版本的匹配文档链接或教程。谢谢
        • 这是一个教程github.com/RichardHightower/boon/wiki/Boon-JSON-in-five-minutes Boon 在公共 maven repo 中。它在 0.27 左右。
        • richardhightower.github.io/site/releases 有 0.4,所以我认为这是最新的。我一直在查看 Boon 的工作项目,它是否有与 Jackson 的 @JsonIgnore 等效的注释?
        • 好处在于 ser/反序列化性能的低端:github.com/fabienrenaud/java-json-benchmark 选择 jackson 或 dsljson 以获得性能。 @RickHigh:我无法在你的 github 上打开问题,如果有任何错误/我错过了,我非常愿意改进我的基准或纠正它的解释。
        【解决方案10】:

        根据输入的 JSON 格式(字符串/文件)创建一个 jSONString。 JSON对应的Message类对象示例如下:

        Message msgFromJSON = new ObjectMapper().readValue(jSONString, Message.class);
        

        【讨论】:

          【解决方案11】:

          JSONObject 转换为Java Object 的简单且有效的java 代码

          Employee.java

          import java.util.HashMap;
          import java.util.Map;
          
          import javax.annotation.Generated;
          
          import com.fasterxml.jackson.annotation.JsonAnyGetter;
          import com.fasterxml.jackson.annotation.JsonAnySetter;
          import com.fasterxml.jackson.annotation.JsonIgnore;
          import com.fasterxml.jackson.annotation.JsonInclude;
          import com.fasterxml.jackson.annotation.JsonProperty;
          import com.fasterxml.jackson.annotation.JsonPropertyOrder;
          
          @JsonInclude(JsonInclude.Include.NON_NULL)
          @Generated("org.jsonschema2pojo")
          @JsonPropertyOrder({
          "id",
          "firstName",
          "lastName"
          })
          public class Employee {
          
          @JsonProperty("id")
          private Integer id;
          @JsonProperty("firstName")
          private String firstName;
          @JsonProperty("lastName")
          private String lastName;
          @JsonIgnore
          private Map<String, Object> additionalProperties = new HashMap<String, Object>();
          
          /**
          *
          * @return
          * The id
          */
          @JsonProperty("id")
          public Integer getId() {
          return id;
          }
          
          /**
          *
          * @param id
          * The id
          */
          @JsonProperty("id")
          public void setId(Integer id) {
          this.id = id;
          }
          
          /**
          *
          * @return
          * The firstName
          */
          @JsonProperty("firstName")
          public String getFirstName() {
          return firstName;
          }
          
          /**
          *
          * @param firstName
          * The firstName
          */
          @JsonProperty("firstName")
          public void setFirstName(String firstName) {
          this.firstName = firstName;
          }
          
          /**
          *
          * @return
          * The lastName
          */
          @JsonProperty("lastName")
          public String getLastName() {
          return lastName;
          }
          
          /**
          *
          * @param lastName
          * The lastName
          */
          @JsonProperty("lastName")
          public void setLastName(String lastName) {
          this.lastName = lastName;
          }
          
          @JsonAnyGetter
          public Map<String, Object> getAdditionalProperties() {
          return this.additionalProperties;
          }
          
          @JsonAnySetter
          public void setAdditionalProperty(String name, Object value) {
          this.additionalProperties.put(name, value);
          }
          
          }
          

          LoadFromJSON.java

          import org.codehaus.jettison.json.JSONObject;
          
          import com.fasterxml.jackson.databind.ObjectMapper;
          
          public class LoadFromJSON {
          
              public static void main(String args[]) throws Exception {
                  JSONObject json = new JSONObject();
                  json.put("id", 2);
                  json.put("firstName", "hello");
                  json.put("lastName", "world");
          
                  byte[] jsonData = json.toString().getBytes();
          
                  ObjectMapper mapper = new ObjectMapper();
                  Employee employee = mapper.readValue(jsonData, Employee.class);
          
                  System.out.print(employee.getLastName());
          
              }
          }
          

          【讨论】:

          • 如何在JSP中访问这些json属性?
          • 如果你不想在 JSP 页面中使用它,那么代码还是一样的。 Employee.java 类将与其相同。但是用 LoadFromJSON.java 编写的代码将被复制到 jsp 页面中,并正确导入每个类。休息根本没有变化。
          【解决方案12】:

          最简单的方法是您可以使用此 softconvertvalue 方法,这是一种自定义方法,您可以在其中将 jsonData 转换为您的特定 Dto 类。

          Dto response = softConvertValue(jsonData, Dto.class);
          
          
          public static <T> T softConvertValue(Object fromValue, Class<T> toValueType) 
          {
              ObjectMapper objMapper = new ObjectMapper();
              return objMapper
                  .configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
                  .convertValue(fromValue, toValueType);
          }
          

          【讨论】:

            猜你喜欢
            • 1970-01-01
            • 2012-08-29
            • 2012-05-29
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            相关资源
            最近更新 更多