Gson教程十二(译):Float和Double类型的特殊值

该文章翻译自Gson Tutorial Series系列教程。该篇主要阐述了Gson如何处理Float和Double类型的特殊值。

上一篇博客中,我们探讨了如何使得JSON的转换具有仁慈性。仁慈性意味着允许JSON某些地方不遵循标准而Gson依然能够解析。在这篇博客中,我们将探究一种允许非标准输入的情况:Gson如何处理Float和Double类型的特殊值(比如**Float.NEGATIVE_INFINITY)。

Float和Double类型的特殊值

在Java中,某些特殊情况下需要将值置为float型和double型。因此,在Java语言刚开始的时候,Float.NEGATIVE_INFINITYFloat.POSITIIVE_INFINITY以及Float.NaN以及他们对应的double型就以及存在了。遗憾的是,JSON标准不知道它们的价值,并没有将它们作为标准的一部分。

让我们引用Gson文档中关于该问题的描述:

JSON规范的2.4章节不允许拥有特殊的double值(NaN,Infinity,-Infinity)。然而,Javascript标准(查看章节4.3.20,4.3.22,4.3.23)允许它们作为Javascript的合法值。甚至,大部分的JavaScript引擎接收这些在JSON中的特殊值而不会发生任何问题。因此,从实践层面来看,即使JSON规范不支持它们,但接收这些值作为合法的JSON也是有理可寻的。

记得吗?在上一篇仁慈性的博客中我们写道,反序列化时Gson对于JSON标准具有很棒的灵活性,而在序列化时又是灵活的。前一部分依然是正确的,但是,这些特殊的float和double值是唯一的意外,它们使得Gson会内在的违反标准。

让我们讲得再透彻一点:如果你传入的JSON使用了一个或多个那样的float或double边缘值,它们会被默认反序列化,而不会有任何问题。

如果你传递一个Float.POSITIVE_INFINITY作为值,Gson将会抛出异常,因为Gson不能根据标准来进行转换。如果你需要传递这些特殊值,你需要使用GsonBuilder来明确的指定这些值可以转换。Gson为你提供了serializeSpecialFloatingPointValues来处理。

是时候来看一个例子了。我们来创建一个用户类,它拥有一个float类型的成员变量weight。

public class UserFloat {  
    String name;
    Float weight;

    public UserFloat(String name, Float weight) {
        this.name = name;
        this.weight = weight;
    }
}

现在,复活节的巧克力兔女郎对我产生了负面影响,最后我的体重大增,因此我需要像下面那样创建我的用户实例:

UserFloat user = new UserFloat("Norman", Float.POSITIVE_INFINITY);

如果你将之传递给Gson,它将会抛出异常:

Gson gson = new Gson();

UserFloat user = new UserFloat("Norman", Float.POSITIVE_INFINITY);

String usersJson = gson.toJson(user); // will throw an exception  

Gson将会给你一个IllegalArgumentException异常,它的陈述如下:

Infinity并不是JSON规范的有效double值。想要使用该行为,使用serializeSpecialFloatingPointValues()方法。

异常信息是非常有帮助的,并且已经提供给了我们解决办法。我们需要使用GsonBuilder

Float.POSITIVE_INFINITY

这将不会有任何异常,并且usersJson的内容是正确的:

GsonBuilder

当然了,这并不是100%的标准,因此你需要确认你的API是否能够处理这些不寻常的值。

前瞻

本篇文章,你学会了Gson是如何处理费标准的JSON float值。默认情况下,Gson不允许序列化这些边缘值。然而,你可以使用GsonBuilder来激活。

下一篇文章,我们将会探讨为你的模型版本化。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值