一、概述
Gson是google提供的用來操作json數據的一個非常好用的類庫。其使用范圍非常的廣泛,所以非常有必要對其進行系統的學習。
json是一種數據格式,確切的說是一種文本數據格式。其在網絡通訊過程中的作用非常的明顯。
目前大多數的網絡通訊格式已經從xml替換為json格式。
其提供了序列化和反序列化的功能。在我們進行網絡開發的過程中通常會把參數封裝成json格式傳給后台,后台解析后的返回結果也會封裝成json格式返回給調用者。
下面就針對json的用法做一下詳細的描述。ps:重要。在查看第三方庫的源代碼或者自己封裝庫的時候尤其重要。對json不熟悉是無法封裝成基於json的高質量庫的。
例如:Retrofit的網絡數據轉換的converter。GsonConverterFactory.create()
二、從小例子入手
ps:Book.java
下面用到的所有實體都是基於Book.java類的。代碼如下:
package com.yw.gsonlib;
import com.google.gson.annotations.SerializedName;
/**
* create by yangwei
* on 2020-02-21 18:08
*/
public class Book {
public Book(){}
private String id;
/**
* 加上@SerializedName注解后服務端api返回name、bookName、Name、bn時,客戶端都能夠拿到正確的值給name屬性。
*/
@SerializedName(value = "bookName",alternate = {"Name","bn"})
private String name;
public Book(String id, String name) {
this.id = id;
this.name = name;
}
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
1.如何創建一個Gson對象
/**
* 創建Gson對象的兩種形式
*/
private void createGsonObject() {
//第一種方式
Gson gson = new Gson();
//第二種方式
Gson gson1 = new GsonBuilder().create();
//方式二除了可以創建一個Gson對象以外還可以進行多項配置,例如,設置日期的格式化
// 例如: new GsonBuilder().setDateFormat("yyyy-MM-dd");
}
2.如何創建一個JsonObject
/**
* 如何創建一個json對象
*/
private void createJsonObject() {
JsonObject jsonObject = new JsonObject();
jsonObject.addProperty("id", "1");//給jsonObject創建一個id屬性值為1
jsonObject.addProperty("bookName", "《深入Java虛擬機》");
jsonObject.addProperty("bookPrice", 36.8);
jsonObject.addProperty("bookColor", "紅色");
//打印Json字符串
System.out.println(jsonObject.toString());
//給JsonObject添加對象
JsonObject jsonObject1 = new JsonObject();
jsonObject1.addProperty("chapterId", "1");
jsonObject1.addProperty("chapterName", "第一章");
//給JsonObject添加實體對象
jsonObject.add("chapter", jsonObject1);
System.out.println(jsonObject.toString());
}
3.數組的序列化與反序列化
/**
* 根據一個json創建一個數組
*/
private void createJsonArray() {
//把數組轉換為Json字符串
String[] args = new String[]{"《深入Java虛擬機》", "《Android插件編程》", "《OpenCV全解》"};
Gson gson = new Gson();
String jsonStr = gson.toJson(args);//先創建一個Json字符串
System.out.println(jsonStr);//輸出
//把Json字符串轉換為Json數組
String[] strArray = gson.fromJson(jsonStr, String[].class);
for (String s : strArray) {
System.out.println(s);
}
}
4.List的序列化與反序列化
/**
* 根據一個json創建一個list
*/
private void createJsonToList() {
//將list集合轉換為json字符串
List<Book> books = new ArrayList<>();
books.add(new Book("1", "《深入Java虛擬機》"));
books.add(new Book("2", "《OpenCV進階》"));
books.add(new Book("3", "《深入Android源代碼》"));
Gson gson = new Gson();
String jsonListStr = gson.toJson(books);
System.out.println(jsonListStr);
//將json字符串轉換為list集合
//獲取泛型的類型
Type type = new TypeToken<List<Book>>() {
}.getType();
//使用gson將字符串轉換為泛型集合,即List<Book>
List<Book> books1 = gson.fromJson(jsonListStr, type);
for (Book book : books1) {
System.out.println(book.getName());
}
}
5.序列化與反序列化綜合例子(基於Book對象)
/**
* json的序列化與反序列化
* 使用toJson實現序列化,使用fromJson實現反序列化
*/
private void jsonSeriaReSeria() {
Gson gson = new Gson();
Book book = new Book("1", "《深入Java虛擬機》");
//將book類序列化成字符串
String bookStr = gson.toJson(book);
System.out.println(bookStr);
//將bookStr反序列化成為Book類
Book b = gson.fromJson(bookStr, Book.class);
System.out.println(b.getName());
}
6.如何為屬性做兼容性處理(重命名、預設名稱等)
/**
* 如何給json屬性重命名
* 假設我們有這樣一個需求:由於接口改版,從服務端返回的api中的book字段的name屬性,變為了bookName。如果我們
* 仍然用name進行解析,則接不到bookName的值,而由於服務端沒有返回name,則解析后name=null。如果要考慮到兼容性那么我們的的Book類的屬性就
* 需要做一下兼容性的處理。
* 使用@SerializedName可以給屬性重命名。用@SerializedName的alternate屬性給Book的屬性做擴展
*/
private void reNameProperty() {
String bookJson = "{\"id\":\"1\",\"bookName\":\"《深入Java虛擬機》\"}";
Gson gson = new Gson();
Book book = gson.fromJson(bookJson, Book.class);
System.out.println(book.getName());
}
7.GsonBuilder的一些基礎配置
/**
* 對GsonBuilder一些屬性介紹
*/
private void gsonBuilderProperty() {
Gson gson = new GsonBuilder()
.serializeNulls()//默認情況下如果某一個屬性為null,那么此屬性不會參與序列化和反序列化的過程,加上此屬性后會參與序列化和反序列化的過程
.setPrettyPrinting()//格式化json字符串的輸出,默認情況下是輸出一行,經過這個屬性設置后會格式化輸出,即有縮進的輸出
.setDateFormat("yyyy-MM-dd HH:mm:ss")//對時間進行格式化
.create();
}
8.TypeAdapter使用方法介紹
/**
* typeAdapter的使用方法
* TypeAdapter是一個泛型抽象類,用於接管某種類型的序列化和反序列化的過程
* 它包含兩個抽象方法write和read,分別用於自定義的序列化和反序列化的過程
*/
private void useTypeAdapter() {
Gson gson = new GsonBuilder().registerTypeAdapter(Book.class,new BookTypeAdapter()).create();
Book book = new Book("1","深入Java虛擬機");
System.out.println(gson.toJson(book));
String bookJson = "{\"Id\":\"1\",\"Name\":\"《深入Java虛擬機》\"}";
Book book1 = gson.fromJson(bookJson,Book.class);
System.out.println(gson.toJson(book1));
}
以下是BookTypeAdapter的代碼介紹
package com.yw.gsonlib;
import com.google.gson.TypeAdapter;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.JsonWriter;
import java.io.IOException;
/**
* create by yangwei
* on 2020-02-23 15:12
*/
public class BookTypeAdapter extends TypeAdapter<Book> {
/**
* 序列化接口
*
* @param jsonWriter
* @param book
* @throws IOException
*/
@Override
public void write(JsonWriter jsonWriter, Book book) throws IOException {
//流式序列化對象開始
jsonWriter.beginObject();
//將Json的key值都指定為首字母大寫
jsonWriter.name("Id").value(book.getId());
jsonWriter.name("Name").value(book.getName());
//流式序列化對象結束
jsonWriter.endObject();
}
/**
* 反序列化接口
*
* @param jsonReader
* @return
* @throws IOException
*/
@Override
public Book read(JsonReader jsonReader) throws IOException {
Book book = new Book();
//流式反序列化開始
jsonReader.beginObject();
while (jsonReader.hasNext()) {
switch (jsonReader.nextName()) {
case "id":
case "Id":
book.setId(jsonReader.nextString());
break;
case "name":
case "Name":
book.setName(jsonReader.nextString());
break;
}
}
//流式反序列化結束
jsonReader.endObject();
return book;
}
}
三、總結
講到這里gson的內容基本上就講完了,相信小伙伴們對gson已經有了一個全面的了解。主要要清楚:如何序列化與反序列化,如何自定義序列化過程、如何更改屬性,如何創建對象、如何對泛型進行序列化與反序列化。
ps:由於后面會講到Retrofit2的數據轉換工廠的實現方式(GsonConverterFactory.create()),對於基礎比較薄弱的開發者來說補充一下Gson相關知識,再接着看Retrofit2的轉換工廠的實現原理會非常的清晰。
另外下一節將介紹自定義泛型,gons和自定義泛型了解清楚后再去看源代碼的實現流程,你會發現根本不需要動腦子,一看就明白了。
