您的位置:首页 > 编程语言 > Python开发

Python 3.3.2 round函数并非"四舍五入"

2013-11-19 01:04 351 查看
对于一些貌似很简单常见的函数,最好还是去读一下Python文档,否则当你被某个BUG折磨得死去活来时,还不知根源所在.尤其是Python这种不断更新的语言.(python 2.7 的round和3.3.2不一样)

3.3.2官方文档对round的定义

round(number[, ndigits])

Return the floating point value number rounded to ndigits digits after the decimal point. If ndigits is omitted, it defaults to zero. Delegates to number.__round__(ndigits).

For the built-in types supporting round(), values are rounded to the closest multiple of 10 to the power minus ndigits; if two multiples are equally close, rounding is done toward the even choice(so, for example, both round(0.5) and round(-0.5) are 0, and round(1.5) is 2). The return value is an integer if called with one argument, otherwise of the same type as number.

举例查看:

>>> round(1.5)
2
>>> round(2.5)
2
>>> round(3.5)
4
>>> round(4.5)
4
>>> round(5.5)
6
>>> round(6.5)
6


分析定义中的关键语句:

values are rounded to the closest multiple of 10 to the power minus ndigits


假设round(123.45, 1)=A,那么A在数轴上是离123.45最近的点(the closest),且A保留1位小数.

"10 to the power minus ndigits"意思是"10的负ndigits次方",即pow(10, -ndigits).

"multiple of n" 表示n的倍数.倍数只能是整数,这个是关键.(可参考维基百科:http://zh.wikipedia.org/wiki/%E5%9B%A0%E6%95%B8)

那么,例如round(123.45, 1)就表示数轴上离123.45最近的"0.1的X倍".

X=1时,"0.1的X倍"(即0.1)是离123.45最近的值吗?显然不是,因为当X=1000时,"0.1的X倍"是100,比0.1更靠近123.45.

按照这个原理进行推导,容易知道当X=1234或1235时,"0.1的X倍"最靠近123.45.那么该取哪个X作为最终结果呢?

if two multiples are equally close, rounding is done toward the even choice


"toward the even choice"意思就是说取偶数.即"0.1的1234倍",即123.4.

然而,当你测试round(123.45, 1):

>>> round(123.45, 1)
123.5


为什么取偶数呢?是BUG吗?非也,官方文档继续解释:

Note

The behavior of round() for floats can be surprising: for example, round(2.675, 2) gives 2.67 instead of the expected 2.68. This is not a bug: it’s a result of the fact that most decimal fractions can’t be represented exactly as a float. See Floating Point Arithmetic: Issues and Limitations for more information

意思就是说计算机需要先将十进制123.45转换为二进制,这个过程会导致二进制的值比123.45略大(比如123.45000001之类的),那么自然就得到123.5这个值了.

这种情形非常普遍:

>>> round(0.05, 1) #should return 0.0
0.1
>>> round(0.15, 1) #should return 0.2
0.1
>>> round(0.25, 1)
0.2
>>> round(0.35, 1) #should return 0.4
0.3
>>> round(0.45, 1) #should return 0.4
0.5
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: