乐趣区

Gson实战之旅

前言

最近因为 FastJson 最近几年的安全漏洞频发,所以后续的项目准备切换到 Gson 中,而前面的项目也开始准备逐步切换到Gson

而本文就是因此而产生,本文将会围绕一系列实际的需求来展示如何使用 Gson 解决实际开发上遇到的问题。

Gson 之旅

如同前言所说的,以下的章节都是本人遇到的实际场景后如何使用 Gson 的记录。

初识 Gson

Gson是一个 Google 开源出来针在 Java 对象和 JSON 数据之间进行相互映射的 Java 类库。而且使用方式很简单。以 maven 项目为例,只需要添加以下配置即可使用。

<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.12</version>
    <scope>provided</scope>
</dependency>

Java 对象转 JSON

而且 GsonAPI使用起来也很简单,例如 Java 对象转成 JSON 字符串,只需要调用 GsontoJson方法即可,示例如下

Gson gson = new Gson();
Student student = new Student();
student.setId(1L);
student.setName("测试用户");
student.setSex(false);
System.out.println(gson.toJson(student));

输出结果

{"id":1,"name":"测试用户","sex":false}

JSON 转 Java 对象

而将 JSON 字符串转成 Java 对象也很简单,只需要调用 GsonfromJson方法即可,示例如下

Gson gson = new Gson();
String jsonStr ="{'id':1,'name':' 测试用户 ','sex':false}";
Student student = gson.fromJson(jsonStr,Student.class);

输出结果

Student(id=1, name= 测试用户, sex=false, birthday=null)

这里需要注意以下两点

  1. fromJson方法需要传入的是 ==JSON 格式的字符串 == 和对应 ==Java 对象的类型 == 才行。
  2. Gson是可以识别使用 == 单引号 == 的 JSON 字符串,而实际上 ==JSON 格式规范里面是需要使用双引号 ==。

演示的 Student 类源码

@Datalombok 框架的注解,可以自动生成对应的 getter/settertoString方法。

@Data
public class Student implements Serializable {
    /**
     * 学生 ID
     */
    private Long id;
    /**
     * 学生名称
     */
    private String name;
    /**
     * 学生性别
     */
    private Boolean sex;
    /**
     * 出生年月日
     */
    private Date birthday;
}

日期格式化

于是就开始了 Gson 的使用,很快我就遇到第一个问题了。使用 Gson 处理第三方接口传过来的 JSON 字符串时,可以正确将 yyyy-MM-dd HH:mm:ss 格式(例如 2020-01-30 11:30:00)的值转为对应的Date 对象。

可是 Java 对象转为 JSON 时却发现 Date 对象的值转为 Thu Jan 30 11:30:00 CST 2020,也就是直接调用了DatetoString方法。而我期待的值则是 2020-01-30 11:30:00。通过查阅用户手册得知,可以使用GsonBuilder 来构建一个自定义日期格式化的 Gson 实例,如下所示

Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd HH:mm:ss").create();
Student student = new Student();
student.setName("测试用户");
student.setId(1L);
student.setSex(false);
student.setBirthday(new Date());
String jsonStr = gson.toJson(student);
System.out.println(gson.toJson(student));

输出结果

{"id":1,"name":"测试用户","sex":false,"birthday":"2020-06-07 18:45:10"}

而针对 JSON 转 Java 对象,常规的日期格式字符(如 yyyy-MM-ddyyyy-MM-dd HH:mm:ss),GSON 都可以正常识别。

显示 null 值

Gson在将 Java 对象转为 JSON 时,会隐藏为值为 null 的属性,但是其实显示所有值为 null 的属性更加便于我们进行调试,而且只要简单地设置一下即可。

Gson gson = new GsonBuilder().serializeNulls().create();
Student student = new Student();
student.setId(1L);
System.out.println(gson.toJson(student));

输出结果

{"id":1,"name":null,"sex":null,"birthday":null}

打印优化

Gson在将 Java 对象转成 JSON 时,默认会进行压缩处理,例如这样

{"id":1,"name":"测试用户","sex":false}

而这个时候只需要简单地设置一下即可让 Gson 输出一个展开的JSON

Gson gson = new GsonBuilder().setPrettyPrinting().create();
Student student = new Student();
student.setId(1L);
student.setName("测试用户");
student.setSex(false);
System.out.println(gson.toJson(student));

输出结果

{
  "id": 1,
  "name": "测试用户",
  "sex": false
}

属性别名

有些时候和第三方系统进行对接时,对方的提供的 JSON 部分属性是某些单词的缩写或者很难理解(遇过有拼音缩写,英语缩写,单词拼写错误),这个时候如果希望自己这边的 Java 对象的属性更好理解,可以选择使用 Gson 的别名功能,示例如下:

@Data
public class Teacher implements Serializable {
    /**
     * 教师姓名
     */
    @SerializedName("name")
    private String teacherName;
    /**
     * 教师性别
     */
    @SerializedName("sex")
    private Boolean teacherSex;
}

@SerializedName注解可以的 value 属性可以设置单个别名,而 alternate 则是用于设置多个别名。

处理泛型

有时候我们会遇到接收一个 JSON 数组的情况,而且此时如果将其转为一个 List 对象时,就会遇到类型转换的问题,以上面 Student 类为例,下面将演示如何将 JSON 数组转成List<Student>

Gson gson = new Gson();
String jsonStr = "[{'name':' 测试用户 '},{'id':2}]";
// 获取实际要转的类型
Type type =new TypeToken<List<Student>>(){}.getType();
List<Student> list = gson.fromJson(jsonStr,type);
System.out.println(list);

输出结果

[Student(id=null, name= 测试用户, sex=null, birthday=null), Student(id=2, name=null, sex=null, birthday=null)]

需要注意这里的 Typejava.lang.reflect.Type

总结

  1. Gson使用 toJson 来讲 Java 对象转为 JSON,而 fromJson 则是将 JSON 转为 Java 对象
  2. 如果默认设置不能满足你的需求,可以选择使用 GsonBuilder 来构建一个自定义的 Gson 对象来进行 JSON 和 Java 对象的互相转换。
退出移动版