一、概述
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和自定义泛型了解清楚后再去看源代码的实现流程,你会发现根本不需要动脑子,一看就明白了。