Gson——嵌套对象的映射

原文链接:Gson — Mapping of Nested Objects

作者:Norman Peitek

翻译:签到钱就到

在Gson博客文章中,我们研究了基本的功能。这篇文章,我们要关注一些更实际的数据,并且研究嵌套对象。你会学会如何简单地处理那些包含复杂数据的对象。

这只是我们Gson系列中的一篇。如果你对其他主题感兴趣,可以看一下下面的大纲:

Gson系列概览

  1. 用java-JSON实现序列化合反序列化
  2. 嵌套对象的映射
  3. Mapping of Arrays and Lists of Objects
  4. Mapping of Maps
  5. Mapping of Sets
  6. Mapping of Null Values
  7. Gson Model Annotations — How to Ignore Fields with @Expose
  8. Gson Model Annotations — How to Change the Naming of Fields with @SerializedName
  9. Gson Builder — Basics & Naming Policies
  10. Gson Builder — Force Serialization of null Values
  11. Gson Builder — Exclusion Strategies
  12. Gson Builder — Relax Gson with Lenient
  13. Gson Builder — Special Values of Floats & Doubles
  14. Gson Builder — Model Versioning
  15. Gson Builder — Formatting of Dates & Custom Date/Time Mapping
  16. Gson Builder — Pretty Printing
  17. Gson Builder — HTML Escaping

嵌套对象的序列化

我们喜欢用实际的例子来演示功能,所以,让我们扩展UserSimple模型。在之前的文章中,user模型只有一些标准的java类型:

public class UserSimple {  
    String name;
    String email;
    boolean isDeveloper;
    int age;
}

现在我们的user也有了一个家庭地址,这个家庭地址也有自己的模型类UserAddress

public class UserNested {  
    String name;
    String email;
    boolean isDeveloper;
    int age;

    // new, see below!
    UserAddress userAddress;
}

public class UserAddress {  
    String street;
    String houseNumber;
    String city;
    String country;
}

换句话说,user现在是用一个UserNested模型表示,其中有一个额外的一对一的地址对象。地址是在UserAddress模型中表示的。

在java中,这两个模型能够通过类清晰地区分,并且我们通过UserAddress userAddress字段保持引用。然而,在JSON里,没有类或引用。只有嵌套nest)用户地址到用户对象中。基本地,在JSON里我们只要用{}在字段名字后创建一个新的对象:

{
    "age": 26,
    "email": "[email protected]",
    "isDeveloper": true,
    "name": "Norman",

    "userAddress": {
        "city": "Magdeburg",
        "country": "Germany",
        "houseNumber": "42A",
        "street": "Main Street"
    }
}

不同于其他属性(ageemail,...),新的userAddress没有直接的值。而是有一些包含在{}内的子值。很重要的一点,需要你理解字段名字后面有括号的就是表明这是一个嵌套对象。

太多的理论。是时候见识一下Gson从UserNested对象中创建了什么。你可能认识这个模式,Gson不需要任何配置。它会自动根据传进的类推断出数据结构:

UserAddress userAddress = new UserAddress(  
    "Main Street", 
    "42A", 
    "Magdeburg", 
    "Germany"
);

UserNested userObject = new UserNested(  
    "Norman", 
    "[email protected]", 
    26, 
    true, 
    userAddress
);

Gson gson = new Gson();  
String userWithAddressJson = gson.toJson(userObject); 

你应该会很好奇,字符串userWithAddressJson的值是什么样子的:

{
    "age": 26,
    "email": "[email protected]",
    "isDeveloper": true,
    "name": "Norman",

    "userAddress": {
        "city": "Magdeburg",
        "country": "Germany",
        "houseNumber": "42A",
        "street": "Main Street"
    }
}

细心的你很容易发现,虽然Gson又一次按照字母顺序存储了字段,但结果正是我们期望的。Gson正确地创建了嵌套userAddressJSON对象。当然,我们可以为用户的薪水方法或工作地址增加更多的嵌套对象。甚至嵌套对象里也能有嵌套对象!

在下一节中,我们会研究另一个方向。如何反序列化复杂的嵌套JSON到Java对象呢?

嵌套对象的反序列化

上一节中,我们假设模型已经有了,并且我们只要创建一个匹配的JSON。特别是对于在真实世界里的开发者,经常是相反的。API接口正在返回一些JSON数据,我们需要为那些数据创建模型类。

如果你已经读了第一篇文章的一些内容,你已经感觉到如何创建一个模型类。防止你觉得无聊,所以我们将从user的例子转移到一个精致的小餐馆。

{
  "name": "Future Studio Steak House",
  "owner": {
    "name": "Christian",
    "address": {
      "city": "Magdeburg",
      "country": "Germany",
      "houseNumber": "42A",
      "street": "Main Street"
    }
  },
  "cook": {
    "age": 18,
    "name": "Marcus",
    "salary": 1500
  },
  "waiter": {
    "age": 18,
    "name": "Norman",
    "salary": 1000
  }
}

这来自于我们的API接口,并且我们要利用Gson去自动创建匹配的Java对象。首先,你需要模型化基础类,所有顶级字段都在里面:

public class Restaurant {  
    String name;

    Owner owner;
    Cook cook;
    Waiter waiter;
}

看一下我们是如何为name创建一个String字符串,以及另外三个扩展的Java类?有些人自已实现了代码,可能与下面的结果不同。事实上,创建一个Java对象并不需要太明确。例如,基于JSON,我们看到了有一样结构的cookwatier嵌套对象。你可以仍然创建一个不同的类,像上面做的一样,或者为他们俩只创建一个普通的Staff类。

public class Restaurant {  
    String name;

    Owner owner;
    Staff cook;
    Staff waiter;
}

其实,两个方法任意一个都可行。如果你还不确定用哪个,我们通常会创建一个额外的类避免未来可能发生的冲突。例如,如果cook模型改变了,但是waiter模型保持不变,你可能需要改变一堆代码。这样,我们暂时抛弃Staff的解决方案。当然,我们仍要为第二层次的对象创建Java模型类:

public class Owner {  
    String name;

    UserAddress address;
}

public class Cook {  
    String name;
    int age;
    int salary;
}

public class Waiter {  
    String name;
    int age;
    int salary;
}

好的,因为这个UserAddress类到哪都能适用,我们作了一点弊,第一部分重复使用了UserAddress。 ;-)

总之,我们希望你理解从一个JSON字符串创建Java模型类的过程。你需要从高层到最深层次遍历,直到你的嵌套JSON只剩下常规类型。

既然主要工作已经完成了,我们可以把一切交给Gson。当然,如果我们之前所做的工作都是正确的,它会优雅地处理一切,并创建只有几行的Java对象:

String restaurantJson = "{ 'name':'Future Studio Steak House', 'owner':{ 'name':'Christian', 'address':{ 'city':'Magdeburg', 'country':'Germany', 'houseNumber':'42', 'street':'Main Street'}},'cook':{ 'age':18, 'name': 'Marcus', 'salary': 1500 }, 'waiter':{ 'age':18, 'name': 'Norman', 'salary': 1000}}";

Gson gson = new Gson();

Restaurant restaurantObject = gson.fromJson(restaurantJson, Restaurant.class);  

restaurantObject对象自动地包含了JSON里的所有的信息 :

提示:从JSON创建Java模型类是一项单调乏味的工作。在你掌握了概念后,你可能想要使用工具自动化处理。我们相当喜欢用jsonschema2pojo.org。

展望

在这篇博客中,我们已经学会了如何用Gson处理嵌套对象。对于Gson序列化复杂的数据类型已经不是问题。甚至反序列化也不是啥大问题,但是你必须首先做一些创建合适模型类的工作。

如果你要反馈或提问,可以在评论里让我们知道,或者twitter@futurestud_io

你可能感兴趣的:(Gson——嵌套对象的映射)