Gson,JSON和LinkedTreeMap的微妙之处

时间:2014-03-09 17:44:54

标签: java json gson

我最近开始使用JSON字符串,并且被告知Google自己的库Gson是处理这些字符串的新方法。

我理解它的方式是JSON字符串本质上是一张地图。每个变量指向字符串中的值。

例如:

String jsonInput2 = "{\"created_at\":\"Sat Feb 08 15:37:37 +0000 2014\",\"id\":432176397474623489\"}

到目前为止,一切都很顺利。创建此JSON字符串的信息可以使用以下代码分配给变量:

Gson gson = new Gson();

Map<String, String> map = new HashMap<String, String>();

map = (Map<String, String>) gson.fromJson(jsonInput, map.getClass());

String createdAt = map.get("created_at");

它在简单的美中几乎是艺术性的。但这就是美丽结束的地方,我的困惑开始了。

以下是上述JSON字符串的扩展名;

String jsonInput2 = "{\"created_at\":\"Sat Feb 08 15:37:37 +0000 2014\",\"id\":432176397474623489\",\"user\":{\"id_str\":\"366301747\",\"name\":\"somethingClever\",\"screen_name\":\"somethingCoolAndClever\"}}";

我的问题是这些“括号内的括号”如何适用于user的{​​{1}}部分?

如何将这些内括号中指定的值分配给变量?

任何人都可以向我解释,或者在代码中向我展示JSON如何处理这样的内容,以及我如何使用它?

简而言之,为什么......

Gson

...打印出String jsonInput = "{\"created_at\":\"Sat Feb 08 15:37:37 +0000 2014\",\"id\":432176397474623489\",\"user\":{\"id_str\":\"366301747\",\"name\":\"somethingClever\",\"screen_name\":\"somethingCoolAndClever\"}}"; Gson gson = new Gson(); Map<String, String> map = new HashMap<String, String>(); map = (Map<String, String>) gson.fromJson(jsonInput, map.getClass()); String name = map.get("name"); System.out.println(name);

5 个答案:

答案 0 :(得分:31)

忘了Java。您需要先了解JSON format。这基本上就是

object
    {}
    { members }
members
    pair
    pair , members
pair
    string : value
array
    []
    [ elements ]
elements
    value 
    value , elements
value
    string
    number
    object
    array
    true
    false
    null

您的第二个JSON String(缺少")如下(使用jsonlint.com进行格式化)

{
    "created_at": "Sat Feb 08 15:37:37 +0000 2014",
    "id": "432176397474623489",
    "user": {
        "id_str": "366301747",
        "name": "somethingClever",
        "screen_name": "somethingCoolAndClever"
    }
}

JSON是一个对象,外部{},包含三对,created_at是一个JSON字符串,id也是一个JSON字符串,user这是一个JSON对象。该JSON对象包含三对,都是JSON字符串。

你问了

  

如何将这些内部括号中指定的值分配给   变量

大多数高级JSON解析/生成库都是为了将​​JSON转换为Pojos并返回。

因此,您可以将JSON格式映射到Java类。

class Pojo {
    @SerializedName("created_at")
    private String createdAt;
    private String id;
    private User user;
}

class User {
    @SerializedName("id_str")
    private String idStr;
    private String name;
    @SerializedName("screen_name")
    private String screenName;
}

// with appropriate getters, setters, and a toString() method

请注意@SerializedName,以便您可以继续使用字段的Java命名约定。

您现在可以反序列化您的JSON

Gson gson = new Gson();
Pojo pojo = gson.fromJson(jsonInput2, Pojo.class);
System.out.println(pojo);

会打印

Pojo [createdAt=Sat Feb 08 15:37:37 +0000 2014, id=432176397474623489", user=User [idStr=366301747, name=somethingClever, screenName=somethingCoolAndClever]]

显示所有字段都已正确设置。

  

任何人都可以向我解释,或者在代码中告诉我,Gson如何处理这样的事情,以及我如何使用它?

Gson的源代码是免费提供的。你可以在网上找到它。它很复杂,源代码解释不适合这里。简而言之,它使用您提供的Class对象来确定它将如何映射JSON对。它查看相应类的字段。如果这些字段是其他类,则它会重新发生,直到它构建了反序列化所需的所有内容的映射。


  

简而言之,为什么...打印出null?

因为您的根JSON对象没有名称为name的对。不要使用Map,而是使用Gson的JsonObject类型。

JsonObject jsonObject = new Gson().fromJson(jsonInput2, JsonObject.class);

String name = jsonObject.get("user")       // get the 'user' JsonElement
                        .getAsJsonObject() // get it as a JsonObject
                        .get("name")       // get the nested 'name' JsonElement
                        .getAsString();    // get it as a String
System.out.println(name);

打印

somethingClever

如果上面的方法类不是正确的类型,它们可能抛出了许多异常。例如,如果我们已经完成了

String name = jsonObject.get("user")       // get the 'user' JsonElement
                        .getAsJsonArray()  // get it as a JsonArray

它会失败,因为user不是JSON数组。具体来说,它会抛出

Exception in thread "main" java.lang.IllegalStateException: This is not a JSON Array.
    at com.google.gson.JsonElement.getAsJsonArray(JsonElement.java:106)
    at com.spring.Example.main(Example.java:19)

因此JsonElement类(JsonObjectJsonArray和其他一些类的父类)提供了检查它的方法。请参阅javadoc。

答案 1 :(得分:5)

JSON字符串具有以下结构:

{
    created_at: "",
    id: "",
    user: {
            id_str: "",
            name: "",
            screen_name: ""
    }
 }

使用代码将值放在地图中时:

Map<String, Object> map = new HashMap<String, Object>();
map = (Map<String, Object>) gson.fromJson(jsonInput, map.getClass());

它具有以下关键值:

created_at
id
user

这就是为什么你可以使用map.get("created_at")

现在,由于您想获取用户的名称,您需要获取用户的地图:

LinkedTreeMap<String, Object> userMap = (LinkedTreeMap<String, Object>) map.get("user");

userMap中,您将获得以下关键值:

id_str
name
screen_name

现在,您可以获得name

user
String name = userMap.get("name");

答案 2 :(得分:4)

对于那些来这里寻找将LinkedTreeMap转换为对象的方法的人来说:

MyClass object = new Gson().fromJson(new Gson().toJson(((LinkedTreeMap<String, Object>) theLinkedTreeMapObject)), MyClass .class)

当我需要解析像:

这样的通用对象时,这对我很有用
Class fullObject {
  private String name;
  private String objectType;
  private Object objectFull;   
}

但我不知道服务器要发送哪个对象。 objectFull将成为LinkedTreeMap

答案 3 :(得分:1)

user本身就是JsonObject

JsonObject user = map.get("user");

答案 4 :(得分:0)

确定。首先,JSON是“JavaScript Object Notation”的缩写,因此您的断言“JSON字符串本质上是一个映射”是不正确的。 JSON块是使用JavaScript语言语法描述的对象图。由于您试图将对象图强制为字符串映射,Sting kay值对,这只适用于任何给定的JSON对象图基本上就是这样的情况(因此不常见)。更成功的策略可能是gson.fromJson(),它会将您的JSON转换为适当的Java对象图。