在很多场景的计算中,最终得到的数值例如123.45678,要截取2位小数得到123.45,而不是默认的四舍五入方法得到123.46,如何实现呢?
文章目录
一.小数点后取2位(四舍五入)的方法
其实这个方法不推荐大家使用,查询资料发现里面的坑其实很多,python2和python3里面的坑还不太一样,在此简单描述一下python3对应的坑的情况。
a = 1.23456
b = 2.355
c = 3.5
d = 2.5
print(round(a, 3))
print(round(b, 2))
print(round(c))
print(round(d))
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
结果:
1.235 # 1.23456最终向前进位了
2.35 # 2.355居然没进位
4 # 最终3.5居然变为4了
2 # 最终2.5取值变为2
- 1
- 2
- 3
- 4
- (1)通过上面的函数,看着是不是很晕,感觉round(x,n)函数是否进位也没看出是啥规律
- (2)round(x,n)函数中,是否进位或四舍五入,取决于n位以及n+1位小数的值
- (3)只有当n+1位数字是5的时候,容易混淆,如果n为偶数,则n+1位数是5,则进位,例如round(1.23456,3)最终变为1.235
- (4)如果n为奇数,则n+1位是数5,那不进位,例如round(2.355,2),最终为2.35
- (5)如果n为0,即没有填写n的时候,最终结果与上面相反,即整数部分为偶数的时候,小数位5不进位,例如(round(2.5)变为2)。
- (6)整数部分为奇数的时候,小数位5进位。(round(3.5)变为4)
看完如上的部分,感觉是不是更晕了,所以round()不推荐使用,目前也不知道设计这个函数的目的在哪里?有谁知道麻烦告知一下?
f = 1.23456
print('%.4f' % f)
print('%.3f' % f)
print('%.2f' % f)
- 1
- 2
- 3
- 4
- 5
结果:
1.2346
1.235
1.23
- 1
- 2
- 3
- (1)原本以为:这个方法是最常规的方法,方便实用,居家旅行必备!
- (2)但是…
f = 0.625
print('%.2f' % f)
# 结果:0.62
- 1
- 2
- 3
- 4
具体是否进位,有个概率问题,感兴趣的朋友可以看看评论里面的信息。
感谢weixin_43094430这位朋友的提示,也感谢其他朋友的参与
from decimal import Decimal
aa = Decimal('5.026').quantize(Decimal('0.00'))
bb = Decimal('3.555').quantize(Decimal('0.00'))
cc = Decimal('3.545').quantize(Decimal('0.00'))
print(aa)
print(bb)
print(cc)
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
结果:
5.03
3.56
3.54
- 1
- 2
- 3
decimal这个模块在很少用,如上图中,3.555结果为3.56,而3.545结果变为3.54,一个5进位了,一个是5没进位,具体原因不详。
所以不推荐使用这个方法!!!
二.小数点后取2位(四舍五不入)的方法
通过计算的途径,很难将最终结果截取2位,我们直接想到的就是如果是字符串,直接截取就可以了。
例如
num = '1234567' #字符串num
print(num[:3])
结果:
123
- 1
- 2
- 3
- 4
- 5
如果是123.456取2位小数(截取2位小数),值需要把小数点右边的当做字符串截取即可
partition()函数(将字符串根据字符串切割):
http://www.runoob.com/python/att-string-partition.html
num = '123.4567'
num_str = num.partition(".")
print(num_str)
结果:
('123', '.', '4567') # 三个元素的元祖
- 1
- 2
- 3
- 4
- 5
- 6
拼接字符串:format()函数的使用
https://blog.csdn.net/i_chaoren/article/details/77922939
def get_two_float(f_str, n):
a, b, c = f_str.partition('.')
c = c[:n]
return ".".join([a, c])
num = "123.4567" #(1)隐患一,传入函数的是字符串
print(get_two_float(num, 2))
num2 = '123.4' # (2)隐患二,如果传入的字符串小数位小于最终取的位数
print(get_two_float(num2, 2))
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
结果:
123.45
123.4
- 1
- 2
最终版本:
def get_two_float(f_str, n):
f_str = str(f_str) # f_str = '{}'.format(f_str) 也可以转换为字符串
a, b, c = f_str.partition('.')
c = (c+"0"*n)[:n] # 如论传入的函数有几位小数,在字符串后面都添加n为小数0
return ".".join([a, c])
num = 123.4567
print(get_two_float(num, 2))
num2 = 123.4
print(get_two_float(num2, 2))
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
结果:
123.45
123.40
round函数的进位方式保证了进位与不进位的概率一样,因为0不存在进位,只有1~9存在,那么如果5固定进位或舍弃的话会导致数据整体偏大或偏小,如此保证了进与舍的值概率相同
对结果尽量取偶数,方便计算和存储。