python 格式化输出 浮点,python怎么把浮点数取整
本文主要介绍Python浮点数舍入、格式化、NaN处理的操作方法。本文详细介绍了三种舍入方法,并详细解释了格式化浮点数输出的示例代码。感兴趣的朋友可以跟随边肖去看一看。
00-1010 1.舍入的三种方法1.1强转换为int类型1.2使用math.ceil和math.floor1.3使用round2。格式化浮点数输出3。执行精确的小数计算4。无穷、负无穷和NaN的判断测验。引用强转换为int类型会直接截断浮点数的小数部分(无论是正数还是负数)。还有一种方法:math.ceil和math.floor不管是正的还是负的,都是这样的:ceil在数轴的正方向上舍入,floor在数轴的负方向上舍入。Round prototype是round(value,ndigits),可以将浮点数四舍五入到固定的小数位。这个函数对正数和负数都采用四舍五入到最近位的原则,当某个值正好是两个整数之间的一半时,四舍五入运算会得到最近的偶数。
目录
1. 取整的三种方法
该方法将直接截断浮点数的小数部分(无论是正数还是负数)。
print(int(2.7)) # 2
print(int(-2.7)) # -2
1.1 强转int类型
该方法的取整规则如下图所示:
可以看出,无论是正还是负,都遵循:ceil沿数轴正方向舍入,floor沿数轴负方向舍入。例子如下:
print(math.ceil(-1.27)) # -1
print(math.floor(-1.27)) # -2
打印(math.ceil(1.27)) # 2
打印(数学地板(1.27)) # 1
1.2 采用math.ceil和math.floor
Round prototype是round(value,ndigits),可以将浮点数四舍五入到固定的小数位。这个函数对正数和负数都采用四舍五入到最近位的原则,当某个值正好是两个整数之间的一半时,四舍五入运算会得到最近的偶数。像1.5和2.5这样的值会四舍五入为2。例子如下:
print(round(1.23,0)) # 1.0
print(round(1.23,1)) # 1.2
print(round(1.27,1)) # 1.3
print(round(-1.27,1)) # -1.3
print(round(1.25361,3)) # 1.254
print(round(1.5,0)) # 2.0
print(round(2.5,0)) # 2.0
传递给round()参数的ndigits可以是负数,在这种情况下,它将被舍入到十、百和千:
a=1627731
打印(round(a,-1)) # 1627730
打印(round(a,-2)) # 1627700
打印(round(a,-3)) # 1628000
1.3 采用round
输出值时,注意不要混淆舍入和格式。如果只输出一个固定位数的数值,一般不需要round(),用format格式化时只需指定需要的精度即可(format()格式化操作会按照round()的规则进行舍入,最后返回一个字符串类型)。
x=1234.56789
s=格式(x, 0.2f )
打印(类型,格式(x,
"0.2f")) # <class str> 1234.57
除了取整到固定小数位,format()
还具有许多格式化功能,如格式化输出对齐,增加千分位分隔符等。实际上面的0.2f
就表示至少对齐到0个字符(相当于没有对齐操作),并保留两位小数。
小提示:.2f
也表示至少对齐到0个字符(默认是0),并保留两位小数,
和0.2f
二者是等效的。
更多示例如下:
# 往右调整以对齐到10个字符print(format(x, ">10.1f")) # 1234.6
# 往右调整以对齐到10个字符
print(format(x, "<10.1f")) # 1234.6
# 居中以对齐到10个字符
print(format(x, "^10.1f")) # 1234.6
# 增加千位分隔符
print(format(x, ",")) # 1,234.56789
# 增加千位分隔符并保存到1位小数
print(format(x, "0,.1f")) # 1,234.6
如果想使用科学计数法,只要把f
改成e
或E
即可:
print(format(x, "e")) # 1.234568e+03print(format(x, "0.2E")) # 1.23E+03
此外,我们还可以利用字符串的translate()
方法交换不同的分隔符:
swap_separators = {ord("."):",", ord(","):"."}print(format(x, ",").translate(swap_separators)) # 1.234,56789
最后,我们这里提一下,调用字符串的.format()
函数和单独调用format()
函数可以达到相同的效果,如:
print("value is {:0.3f}".format(x)) # value is 1.235print("The value is {:0,.2f}".format(x)) # The value is 1,234.57
当然我们也可以使用%
操作符来对数值做格式化处理,如:
print("%.2f" % x)print("%10.1f" % x)
print("%-10.1f" % x)
这种格式化操作虽然可行,但是比起更加现代化的format()
方法,这种方法就显得不是那么强大了。如用%
操作符来格式化数值时,有些功能就没法得到支持了(如添加千位分隔符)。
3. 执行精确的小数计算
我们在第一部分介绍了round()
函数,我们有可能会企图用浮点取整的方式来修正精度上的问题,如:
a = 2.1b = 4.2
c = a + b
print(c) # 6.300000000000001
print(c==6.3) # False
print(round(c, 2)) # 6.3 企图这样修正精度(???)
对大部分浮点数应用程序(包括科学计算与机器学习)来说,一般都不必(或者所不推荐)这么做。虽然Python的浮点运算会引入一些小误差,但这些误差实际上是底层CPU的浮点运算单元和IEEE 754浮点算数标准的一种特性。由于Python的浮点数类型保存的数据采用的是原始保存形式,因此只要代码中用到了float
实例,那就无法避免这样的误差。
如果避免出现误差的行为非常重要(比如在金融应用中),那么可以考虑使用decimal
模块。事实上在用Python做数据库库接口时经常碰到Decimal
对象——当访问金融数据时尤其如此。我们通过使用Decimal
对象解决上述问题:
from decimal import Decimala = Decimal(4.2)
b = Decimal(2.1)
print(type(a + b), a + b) # <class decimal.Decimal> 6.3
print((a + b) == Decimal(6.3)) # True
这么做看起来似乎有点怪异(将数字以字符串的形式来指定)。但是Decimal
对象能够以任何期望的方式来工作(支持所有常见的数学操作)。如果要将它们打印出来或者在字符串格式化函数中使用,它们看起来就和普通数字一样。它们也可以和普通int
、float
类型混合操作(最后会统一强转为Decimal
类型):
print(type(a + 1), a + 1) # <class decimal.Decimal> 5.2
但是需要注意的是不要将其与普通float
类型直接进行比较:
print((a + b) == 6.3) # False
decimal
模块的强大之处在于在计算过程中灵活地控制数字的位数和四舍五入,如我们可以创建一个本地的上下文环境然后修改精度的设定,如:
from decimal import localcontexta = Decimal("1.3")
b = Decimal("1.7")
print(a/b) # 0.7647058823529411764705882353
with localcontext() as ctx:
ctx.prec = 3
print(a/b) # 0.765
with localcontext() as ctx:
ctx.prec = 50
print(a/b) # 0.764705882352941176470588235294117647058823529
不过还是我们上面所说的,如果我们处理的是科学或工程类型的问题,那么更常见的做法是直接使用普通的float
浮点类型。首先,在真实世界中极少有东西需要计算到小数点后17位(float
提供17位的精度),因此在计算中引入的微小误差不足挂齿;其次,原生的float
浮点数运算性能要快许多——如果要执行大量计算,性能问题就显得很重要了。
在使用float
类型时,我们同样还需要对类似相减抵消(substraction cancellation)以及把大数和小数加载一起的情况多加小心:
nums = [1.23e+18, 1, -1.23e+18]print(sum(nums)) # 0.0
使用Decimal
对象当然可以解决此问题。不过在不动用Decimal
对象的情况下,我们可以使用math.fsum()
以更精确的实现来解决:
import mathprint(math.fsum(nums)) # 1.0
但对于其它复杂的数值算法,我们就需要研究算法本身,理解其误差传播(error propagation)了,这属于数值分析的研究范畴。在数值分析中数学家研究了大量数值算法,其中一些算法的误差处理能力优于其它算法,详情可以参见我的数值计算专栏《orion-orion:数值计算》,此处不再详述。
4. 无穷大、负无穷大和NaN的判断测试
在实际项目中我们需要对浮点数的无穷大、负无穷大或NaN(not a number)进行判断测试。在Python中没有特殊的语法来表示这些特殊的浮点值,但是它们可以通过float
来创建:
a = float("inf")b = float("-inf")
c = float("nan")
print(a, b, c) # inf -inf nan
要检查是否出现了这些值,可以使用math.isinf()
和math.isnan()
函数:
print(math.isinf(a)) # Trueprint(math.isnan(c)) # True
这些特殊浮点数的详细信息可以参考IEEE 754规范。但是我们这里有几个棘手的问题需要搞清楚,尤其是设计比较操作和操作符时可能出现的问题。
无穷大值在数学计算中会进行传播,如:
a = float("inf")print(a + 45) # inf
print(a * 10) # inf
print(10/a) # 0.0
但是,某些关于无穷大值特定的操作会导致未定义的行为并产生NaN
的结果,例如:
a = float("inf")print(a/a) # nan
b = float("-inf")
print(a + b) # nan
NaN会通过所有的操作进行传播,且不会引发任何异常,如:
c = float("nan")print(c + 23) # nan
print(c / 2) # nan
print(c + 2) # nan
有关NaN,一个微妙的特性是他们在做比较时从不会被判定为相等,如:
c = float("nan")d = float("nan")
print(c == d) # False
print(c is d) # False
正因为如此,唯一安全检测NaN的方法是使用math.isnan()
。
参考
[1] Martelli A, Ravenscroft A, Ascher D. Python cookbook[M]. " O'Reilly Media, Inc.", 2015.
[2] https://stackoverflow.com/questions/15765289/what-is-the-difference-between-0-2lf-and-2lf-as-printf-placeholders
[3] https://docs.python.org
到此这篇关于Python浮点数取整、格式化和NaN处理的操作方法的文章就介绍到这了,更多相关Python浮点数取整内容请搜索盛行IT软件开发工作室以前的文章或继续浏览下面的相关文章希望大家以后多多支持盛行IT软件开发工作室!
郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。