Gson:GitHub 标星 18K 的 JSON 解析器,Google 出品的 Java JSON 解析器,强烈推荐!

Gson gson = new Gson();

int one = gson.fromJson(“1”, int.class);

Integer two = gson.fromJson(“2”, Integer.class);

Boolean false1 = gson.fromJson(“false”, Boolean.class);

String str = gson.fromJson(““王二””, String.class);

String[] anotherStr = gson.fromJson(“[“沉默”,“王二”]”, String[].class);

System.out.println(one);

System.out.println(two);

System.out.println(false1);

System.out.println(str);

System.out.println(Arrays.toString(anotherStr));

toJson() 方法用于序列化,对应的,fromJson() 方法用于反序列化。不过,你需要在反序列化的时候,指定参数的类型,是 int 还是 Integer,是 Boolean 还是 String,或者 String 数组。

来看一下输出结果:

1

2

false

王二

[沉默, 王二]

上面的例子都比较简单,还体现不出来我的威力。

下面,我们来自定义一个类:

public class Writer {

private int age = 18;

private String name = “王二”;

private transient int sex = 1;

}

然后,我们来将其序列化:

Writer writer = new Writer();

Gson gson = new Gson();

String json = gson.toJson(writer);

System.out.println(json);

用法和之前一样简单,来看一下输出结果:

{“age”:18,“name”:“王二”}

同样,可以将结果反序列化:

Writer writer1 = gson.fromJson(json, Writer.class);

这里有一些注意事项,我需要提醒你。

1)推荐使用 private 修饰字段。

2)不需要使用任何的注解来表明哪些字段需要序列化,哪些字段不需要序列化。默认情况下,包括所有的字段,以及从父类继承过来的字段。

3)如果一个字段被 transient 关键字修饰的话,它将不参与序列化。

4)如果一个字段的值为 null,它不会在序列化后的结果中显示。

5)JSON 中缺少的字段将在反序列化后设置为默认值,引用数据类型的默认值为 null,数字类型的默认值为 0,布尔值默认为 false。

接下来,来看一个序列化集合的例子。

List list =new ArrayList<>();

list.add(“好好学习”);

list.add(“天天向上”);

String json = gson.toJson(list);

结果如下所示:

[“好好学习”,“天天向上”]

反序列化的时候,也很简单。

List listResult = gson.fromJson(json,List.class);

结果如下所示:

[好好学习, 天天向上]

我女朋友是一个很细心也很贴心的人,在你调用 toJson() 方法进行序列化的时候,她会先判 null,防止抛出 NPE,再通过 getClass() 获取参数的类型,然后进行序列化。

public String toJson(Object src) {

if (src == null) {

return toJson(JsonNull.INSTANCE);

}

return toJson(src, src.getClass());

}

但是呢?对于泛型来说,getClass() 的时候会丢掉参数化类型。来看下面这个例子。

public class Foo {

T value;

public void set(T value) {

this.value = value;

}

public T get() {

return value;

}

public static void main(String[] args) {

Gson gson = new Gson();

Foo foo = new Foo();

Bar bar = new Bar();

foo.set(bar);

String json = gson.toJson(foo);

}

}

class Bar{

private int age = 10;

private String name = “图灵”;

}

假如你 debug 的时候,进入到 toJson() 方法的内部,就可以观察到。

foo 的实际类型为 Foo<Bar>,但我女朋友在调用 foo.getClass() 的时候,只会得到 Foo,这就意味着她并不知道 foo 的实际类型。

序列化的时候还好,反序列化的时候就无能为力了。

Foo foo1 = gson.fromJson(json, foo.getClass());

Bar bar1 = foo1.get();

这段代码在运行的时候就报错了。

Exception in thread “main” java.lang.ClassCastException: class com.google.gson.internal.LinkedTreeMap cannot be cast to class com.itwanger.gson.Bar (com.google.gson.internal.LinkedTreeMap and com.itwanger.gson.Bar are in unnamed module of loader ‘app’)

at com.itwanger.gson.Foo.main(Foo.java:36)

默认情况下,泛型的参数类型会被转成 LinkedTreeMap,这显然并不是我们预期的 Bar,女朋友对此表示很无奈。

作为 Google 的亲儿子,我的血液里流淌着“贵族”二字,我又怎能忍心女朋友无助时的落寞。

于是,我在女朋友的体内植入了另外两种方法,带 Type 类型参数的:

toJson(Object src, Type typeOfSrc);

T fromJson(String json, Type typeOfT);

这样的话,你在进行泛型的序列化和反序列化时,就可以指定泛型的参数化类型了。

Type fooType = new TypeToken<Foo>() {}.getType();

String json = gson.toJson(foo,fooType);

Foo foo1 = gson.fromJson(json, fooType);

Bar bar1 = foo1.get();

debug 进入 toJson() 方法内部查看的话,就可以看到 foo 的真实类型了。

fromJson() 在反序列化的时候,和此类似。

这样的话,bar1 就可以通过 foo1.get() 到了。

瞧,我考虑得多周全,女朋友都忍不住夸我了!

05、处理混合类型

你知道的,Java 不建议使用混合类型,也就是下面这种情况。

List list = new ArrayList();

list.add(“沉默王二”);

list.add(18);

list.add(new Event(“gson”, “google”));

Event 的定义如下所示:

class Event {

private String name;

private String source;

Event(String name, String source) {

this.name = name;

this.source = source;

}

}

由于 list 没有指定具体的类型,因此它里面可以存放各种类型的数据。这样虽然省事,我女朋友在序列化的时候也没问题,但反序列化的时候就要麻烦多了。

Gson gson = new Gson();

String json = gson.toJson(list);

System.out.println(json);

输出结果如下所示:

[“沉默王二”,18,{“name”:“gson”,“source”:“google”}]

反序列化的时候,就需要花点心思才能拿到 Event 对象。

JsonParser parser = new JsonParser();

JsonArray array = parser.parse(json).getAsJsonArray();

String message = gson.fromJson(array.get(0), String.class);

int number = gson.fromJson(array.get(1), int.class);

Event event = gson.fromJson(array.get(2), Event.class);

承认了,JsonParser 是我的前任。希望你不要喷我渣男,真不是我花心,是因为我们性格上有些不太适合。但我们仍然保持着朋友的关系,因为我们谁都没有错,只是代码更加规范了,已经很少有开发者使用混合类型了。

06、个性化定制

考虑到你是一个追求时髦的人,我一直对自己要求很高,力争能够满足你的所有需求。这种高标准的要求,让我女朋友对我是又爱又恨。

爱的是,我这种追求完美的态度;恨的是,她有时候力不从心,帮不上忙。

使用 toJson() 序列化 Java 对象时,返回的 JSON 字符串中没有空格,很紧凑。如果你想要打印更漂亮的 JSON 格式,你需要打电话给一个叫 GsonBuilder 的老板,让他进行一些定制,然后再把复刻版邮寄给你,就像我在使用指南中提到的那样。

public class Writer {

private int age = 18;

private String name = “沉默王二”;

public static void main(String[] args) {

Writer writer = new Writer();

Gson gson = new Gson();

String json = gson.toJson(writer);

System.out.println(json);

Gson gson1 = new GsonBuilder().setPrettyPrinting().create();

String jsonOutput = gson1.toJson(writer);

System.out.println(jsonOutput);

}

}

来对比一下输出结果:

{“age”:18,“name”:“沉默王二”}

{

“age”: 18,

“name”: “沉默王二”

}

通过 setPrettyPrinting() 定制后,输出的格式更加层次化、立体化,字段与值之间有空格,每个不同的字段之间也会有换行。

之前提到了,默认情况下,我女朋友在序列化的时候会忽略 null 值的字段,如果不想这样的话,同样可以打电话给 GsonBuilder。

public class Writer {

private int age = 18;

private String name = null;

public static void main(String[] args) {

Writer writer = new Writer();

Gson gson = new Gson();

String json = gson.toJson(writer);

System.out.println(json);

Gson gson2 = new GsonBuilder().serializeNulls().create();

String jsonOutput2 = gson2.toJson(writer);

System.out.println(jsonOutput2);

}

}

来对比一下输出结果:

{“age”:18}

{“age”:18,“name”:null}

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
结果:

{“age”:18}

{“age”:18,“name”:null}

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

[外链图片转存中…(img-Caoss7zQ-1715688537332)]

[外链图片转存中…(img-7GwBCWOU-1715688537333)]

[外链图片转存中…(img-renvdpge-1715688537333)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 20
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值