Representation of Integers and Reals
Section II
【原文见:http://www.topcoder.com/tc?module=Static&d1=tutorials&d2=integersReals2】
作者 By misof
TopCoder Member
翻译 农夫三拳@seu
[email protected]
流言: 浮点数变量不仅仅可以存储数字还可以存储一些异值
正确性: 正确
正如前面解答所说,标准中保留了指数的最大值和最小值来存储特殊的数。(注意在内存中这些指数的值分别存储为“全0”和“全1”。)
零(Zero)
当我们谈起符号-尾数-指数的表示时,实际上指的是任何非0数字可以用此方式进行表示。0并不能用这种方式直接表示。为了表示0,我们使用指数和尾数部分全0这样一个特殊的值来表示。注意-0和+0是两个不同值,尽管他们比较时是相等的。
这里有必要注意一下如果memset()用0来对一个浮点数的数组进行填充,那么存储的数将为0。同样的,由于C++中的全局变量初始化为0的位形式,因此全局的浮点数变量也被初始化为0。
这里有关于负0的相当一部分狡猾的陷阱。例如,如果x=0.0,表达式"0.0 - x"和"-x"并不是等价的,前面表达式的值是0.0,后面则是-0.0。
我对这个问题的看法是:负0能够“制造一个学习体验的机会”,当他们像以往一样打印出"-0"或者"-0.0"(这里“学习体验”指的是你花费在学习为什么得到这些奇怪值的时间和精力)。
无穷数(Infinities)
正无穷大和负无穷大对应于指数全1和尾数全0。符号位决定负无穷和正无穷。能够将无穷大表示成一个值是非常有用的因为
它们允许操作能够在越过溢出的情况下继续执行。
不是一个数字(Not a Number)
值NaN(Not a Number)被用来表示一个值不是实数。NaN被表示成一个指数全1,尾数非0的位形式。有两类NaN: QNaN (Quiet NaN) and SNaN (Signaling NaN)。
QNaN指的是尾数设置了最高有效位的NaN。QNaN通常出现在数学运算中。当结果在数学中没有定义时,通常产生这些QNaN值。(例如,3*sqrt(-1.0)就是一个QNaN)
SNaN指的是尾数清除了最高有效位的NaN。它用来标记操作时引发的异常。SNaN可以很容易的被指派给一个未初始化的变量来捕获一些不正确的使用。
【译者注:最高有效位(MSB) -- 指二进制中最高值的比特。例如,在十进制的15389这一数字中,相当于万位上的数字(1)对数值的影响最大。比较与之相反的“最低有效位”(LSB)。】
如果一个返回值是QNaN,那么意味着无法得知操作的结果,而SNaN则意味着操作本身是非法的。
低能数(Subnormal numbers)
我们始终没有使用指数全0和尾数非0的情况。我们有这些值存储和0非常近的数字。
这些数字被叫做“低能的”,因为他们比通常能够表示的数字要小。这里我们假定二进制小数点前面开头没有1。如果符号位是s的话,指数部分全0并且尾数为m,那么这个存储的值为(-1)^s*0.m*2^(-q),这里对于单精度数q是126,对于双精度数是1022。(注意0仅仅是低能数的一个特例。但我们仍然将它独立表示。)
总结所有可能的值(Summary of all possible values)
在下面的表中,b是用来存储指数部分时的偏移量,也就是..单精度时是127,双精度时是1023.
符号位s | 指数e | 尾数m | 可表示的数 |
0 | 00...00 | 00...00 | +0.0 |
0 | 00...00 | 00...01 to 11...11 | 0.m x 2-b+1 |
0 | 00...01 to 11...10 | anything | 1.m x 2e-b |
0 | 11...11 | 00...00 | +Infinity |
0 | 11...11 | 00...01 to 01...11 | SNaN |
0 | 11...11 | 10...00 to 11...11 | QNaN |
1 | 00...00 | 00...00 | -0.0 |
1 | 00...00 | 00...01 to 11...11 | -0.m x 2-b+1 |
1 | 00...01 to 11...10 | anything | -1.m x 2e-b |
1 | 11...11 | 00...00 | -Infinity |
1 | 11...11 | 00...01 to 01...11 | SNaN |
1 | 11...11 | 10...00 to 11.11 | QNaN |
对所有特殊的数进行操作(Operations with all the special numbers)
所有上述能够表示的特殊数字都是定义好的。这意味着你的程序不会仅仅因为某一个计算出来的值超过可以表示的范围而崩溃。但这仍然是一个不期望看到的情况,如果它发生了,你应该在你的程序中进行检查并且当它出现时进行处理。
所有的操作都是以直觉的方式定义的。任何有NaN参与的 操作将会产生NaN的结果。一些其他的操作见下表。(在表中,r是一个可以被表示的正数。∞是无穷大数,÷是平常的浮点数除法。)一个完整的列表可以在标准中或者你的编译器文档中查找到。注意甚至于这些值的比较操作都进行了定义。这个话题超过了本篇文章,如果你感兴趣的话,可以浏览文章末尾的参考。
操作 | 结果 |
0 ÷ ±∞ | 0 |
±r ÷ ±∞ | 0 |
(-1)s∞ x (-1)t∞ | (-1)st∞ |
∞ + ∞ | ∞ |
±r ÷ 0 | ±∞ |
0 ÷ 0 | NaN |
∞ - ∞ | NaN |
±∞ ÷ ±∞ | NaN |
±∞ x 0 | NaN |
当比较位模式的时候,前面的一些位组成指数,指数越大,位模式的字典序就越靠后。类似的,具有相同指数的位模式可以通过比较它们的尾数。
另外可以考虑成这样:当比较两个以上述形式存储的非负数的时候,比较的结果总是同具有相同位模式的整数的比较一致。(注意这将使得比较相同的快)
流言: 比较浮点数相等通常是一个坏主意
正确性: 正确
考虑下面的代码:
将打印多少个星呢?10个?运行它看看,结果将是令人惊讶的。这个代码持续不断的打印星直到我们结束它。 问题在哪呢?正如我们所知道的,双精度数并不是无限精确的。我们在这里碰到的问题是这样的:在二进制中,0.1的表示并不是有限的。十进制的0.1和二进制的0.0(0011)是等价的,这里括号括住的部分是循环部分。当0.1存储在double变量中,它被近似成最近的可以表示的数。因此当我们加上10次之后,结果并不是刚刚好等于1。
最常见的建议是当比较两个双精度的数时,使用一些误差(通常用ε表示)。例如,你也许听到如下的提示:“考虑双精度数a和b相等时使用:if(fabs(a-b)<1e-7)”。注意这个只是一小步提高而不是最佳的方法。后面我们将给出一个更好的方法。
单精度数通过类似的计算显示他们能够仅能够存储最多7个最高有效位的十进制数。这对于实际应用来说太少了,并且最重要的是,它提供的精度要小于TC在要求返回浮点数值时所要求的精度。道理非常清楚:绝对不要使用单精度数!严重一点的说,想都不要想。现在都是有足够的可用内存了。
附带一点注意,一旦近似的错误在你的计算中出现,它们将会引起更深的计算错误。因此即使结果应该是一个整数,它在浮点数中的表示也不是一模一样的。可以想想上面打印星号的循环那个例子。
流言: 我听说long double能提供更好的精度
正确性: 平台相关
扩展双精度类型是一个不常使用的IEEE-754标准定义的精度类型,它需要79位的存储空间。一些编译器有一个对应这个精度的数据类型,有一些则没有。例如,在g++中x86架构的机器上有一个使用10个字节(80个位)内存的数据类型long double。(在MSVC++中,有long double类型,但是它被映射成了double类型。)
80个位的扩展双精度类型在Intel 80*87双核浮点运算中内部使用,其目的是位了能够在来来回回的位移操作中不丢失IEEE-754标准下64位(和32位)格式的精度。当g++中的优化被设置为非0值时,g++在生成代码时将在内部使用long double而不是double和single。这个扩展形式能够存储19位10进制的最高有效位。
如果你需要更加高的精度,你要么实现自己的数学运算,或者使用Java数学库中的BigInteger和BigDecimal类。
这样的讨论的例子还有: here, here, here, here 和 here.(它们都是值得一读的,从其他人的错误中学习要比从自己当中少痛苦一点)
我们将通过展示另外一个简单的例子来开始我们的解答。
这个程序将会打印多少个点呢?这次很清楚,不是吗?这次的终止条件不再是等于测试。这个循环将会在10^22遍历后停止。或者。。是吗?
真糟糕,这又一次的成了死循环。为什么会这样呢?因为当r这个值变得大了,这个变量的精度并不能够足够大的存储所有r的十进制下的数。最后一些会丢失掉。因此当我们在这么大的一个数后加1,结果又被近似到原来的数了。
练习:试着判断一下我们这个循环里r所能达到的最大值。检查你的答案,如果你的判断出错了,找出它的原因。
在作出观察之后,我们将显示为什么表达式fabs(a-b)<epsilon(使用固定值epsilon,推荐在1e-7到1e-9)对于比较双精度数并不是理想的。
考虑值123456123456.1234588623046875和123456123456.1234741210937500。它们都没什么特别之处,仅仅是两个double可以在不进行近似的情况下就能存储的两个值。他们直接的差大约在2e-5.
现知让我们看看这两个值的位形式:
是这样的,没错。这是在double中可以被存储的两个连续的值。任何使用近似得到的错误都能够使得其中对一个变成另外一个(或者超过)。但是他们仍然是不一样的,因此我们原始的测试“相等”不能够工作。
我们真正需要的是容忍一些小的精度误差。正如我们所看到的,double能够近似存储最多15个10进制数字的数。通过近似积聚的精度错误,最后一些数字将会丢失。但是我们究竟应该怎样容忍这些误差呢?
我们将不使用固定常数ε,而使用一个与比较的数数量级相关的值。更确切点说,如果x是一个双精度数,那么x*1e-10是一个比x小10倍数量级的数。它的最高有效位对应于x的第11位最高有效位。这使得它能够很好的满足我们的需要。
换句话说,一个更好的方式来比较a,b两个双精度数是否“相等”就是检查a是否在b*(1-1e-10)和b*(1+1e-10)之间。(小心,如果b是负数的时候,这两个数中的第一个将会更大!)
看到用这样的比较方式的问题了吗?试着比较1e-1072和-1e1072.这两个数都机会相等并且等于0,但是我们的测试在处理这种情况时会失败。这就是为什么我们既需要做第一个测试(测试绝对误差)和第二个测试(测试相对误差)。
这就是TC中用来检查你的返回值是否正确的方法。现在你知道原因了。
有更好的比较函数(参见其中的一篇参考文章),但是更重要的是要知道在实际中你经常仅仅使用绝对误差测试而侥幸成功。为什么?
因为包含在计算当中的数字都是在限定的范围之内。例如,如果我们需要比较的最大数只是9947,那么你知道一个double能够在十进制小数点后存储另外的11个数字。因此我们在进行绝对误差测试时使用epsilon=1e-8,我们运行最后的3个数字丢掉。
这个方法的优点很明确:检查绝对误差要比上面的高级测试简单。
作为一个有用的例子,注意如果一个整数n是平方数(也就是说,对于某个整数k,n=k^2),那么sqrt(double(n))将返回k的精确值。正如我们所知道k可以存储为与n相同类型的变量,代码 int k = int(sqrt(double(n)))是安全的,这里将不会出现近似错误。
流言: 如果我将同样的计算做两次,那么得到的结果将相等
正确性: 部分正确
等等,仅仅是部分正确?这个不是与上面的解答矛盾了吗?当然,其实并不矛盾。
在C++当中,这个流言并不总是正确。问题在于根据标准一个C++编译器有的时候能够使用一个更大的数据类型进行内部运算。并且事实上,g++有的时候内部使用long double而不是double来得到较高的精度。存储的值只有在必要时才转换类型为double。如果是由你的编译器来决定你计算实例是内部使用long double还是double,那么不同的近似将影响结果,因此最终的结果也就不一样了。
这是一个几乎无法发现的bug并且非常烦人。假设你在你计算中的每一步加入了一个调试输出,你无意中是的每一步之后中间结果都被转换成了double并且输出。换句话说,你迫使编译器在内部使用double,瞬间一切都能工作正常了。但是当你移去这些调试输出时,这个程序又开始出错了。
一种方式是写代码时仅使用long double。
不幸的是,这个只能解决许多可能问题中的一个。另外一个是当编译器优化你的代码时将计算操作的顺序重新进行了安排。在不同的场合下,两个相同的C++代码片段得到了不同的两段指令。然后所有的精度问题又都回来了。
作为一个例子,表达式x+y-z可能一次被看成x+(y-z),而另外一次可能变成了(x+y)-z. 试试当x=1.0,y=z=10^30的情况。
因此即使你有两段相同的代码,你也不能保证他们会有相同的结果。如果你想要这个保证的话,将这些代码封装成一个函数然后在这两种情况下调用同样的函数。
Further reading