Gson使用方法详解

一、概述

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

  

 

上一篇:Gson源码解析之-fromJson


下一篇:判断字符串是否是合法字符串