首先了解下round函数,很简单,对浮点数进行近似取值,保留几位小数
用法:
round(number,ndigits)
number:要四舍五入的数,ndigits:要小数点后保留的位数
在实际使用中发现round函数并不总是如上所说的四舍五入。如:
In [14]: round(2.355, 2)
Out[14]: 2.35
py2.x和py3.x版本运行结果
>>> round(2.675, 2)
2.67
简单的说就是,round(2.675, 2) 的结果,不论我们从python2还是3来看,结果都应该是2.68的,结果它偏偏是2.67,为什么?这跟浮点数的精度有关。我们知道在机器中浮点数不一定能精确表达,因为换算成一串1和0后可能是无限位数的,机器已经做出了截断处理。那么在机器中保存的2.675这个数字就比实际数字要小那么一点点。这一点点就导致了它离2.67要更近一点点,所以保留两位小数时就近似到了2.67。
由于round函数是四舍六入五平分的机制问题,所以想要得到传统的四舍五入,使用round函数是不准确的,于是自己编写了一个函数
def round_up(number, ndigits:int=0):
"""模拟round内置函数
此函数不再是四舍六入五平分的规则
如果ndigits=0时,根据内置round函数取整数,所以ndigits尽量不要使用默认值
例如 用number=1.3447543的例子,比如保留2位
保留2位时,要保证前3位的值不能变,因为要对第3位做判断是否大于5,
但是直接取3位小数,那么第4位的值会对第3位的值产生影响,比如:1.3447,保留3位会变成 1.345,对结果有影响,
所以直接保留4位,第四位的值怎么变化都无所谓,因为我们不用,只用前3位
: param number: 保留的值
:param ndigits: 保留的位数
"""
try:
if ndigits == 0:
# %.0f ---> 不保留小数,这种取值也会四舍六入五平分
res_3 = f"%.{ndigits}f" % number
print(res_3)
elif ndigits > 0:
# 多保留2位number的小数 因为自带的保留值是不准确的(%.2f) ---->相当于保留了4位
res_3 = f"%.{ndigits+2}f" % number
print(res_3) # 1.3448
# 以"."分割开整数和小数,["1","3448"], 取小数位------>>>除以10之后取整得:344
res_2 = int(float(res_3.split(".")[-1]) / 10)
# 得到余数,后面根据余数来判断是否进1 344 % 10 = 4,其中商为34是保留的位数
res_1 = res_2 % 10
# pow计算乘方(10的ndigits方)
# 得到保留位数的值,然后在判断是否进"1"
last_res = int(number) + (int(res_2 / 10)) / pow(10, ndigits)
if res_1 < 5:
print(last_res)
else:
# 保留的位数进“1”
last_res = last_res + 1/pow(10, ndigits)
# 防止出现相加后得:1.324999999999999
print(f"%.{ndigits}f" % last_res)
except TypeError:
print("输入的内容不是数字")
round_up(2.675, 2) ===> 2.68
print(round(2.675,2)) ===> 2.67