Вы можете использовать Десятичный модуль для настраиваемых округлений :
import decimal
def rnd(f,n,r=decimal.ROUND_HALF_DOWN):
pat='1.'+'0'*n
return str(decimal.Decimal(str(f)).quantize(decimal.Decimal(pat),r))
Теперь для проверки:
for n in (500.99,100.39,5.019):
for i in (0,1,2):
print('n={:10}, i={:2}, Down: {:>10}, Closest: {:>10}'.format(n,i, rnd(n,i,decimal.ROUND_FLOOR),rnd(n,i)))
Отпечатки:
n= 500.99, i= 0, Down: 500, Closest: 501
n= 500.99, i= 1, Down: 500.9, Closest: 501.0
n= 500.99, i= 2, Down: 500.99, Closest: 500.99
n= 100.39, i= 0, Down: 100, Closest: 100
n= 100.39, i= 1, Down: 100.3, Closest: 100.4
n= 100.39, i= 2, Down: 100.39, Closest: 100.39
n= 5.019, i= 0, Down: 5, Closest: 5
n= 5.019, i= 1, Down: 5.0, Closest: 5.0
n= 5.019, i= 2, Down: 5.01, Closest: 5.02
(Примечание. Это разный выход в случае 1 десятичного знака и ввод 100.39
. Пожалуйста, уточните желаемый вывод 100.2
в этом случае? Разве вывод не должен быть 100.3
?)
С уточнением, я думаю, вам просто нужно использовать fmod
:
import math
def rnd(n,f,prec,direction='down'):
if direction=='down':
if prec==0:
return str(int(n))
return str(round(n-math.fmod(n,f),prec))
else:
if prec==0:
return str(int(round(n)))
n+=f
return str(round(n-math.fmod(n,f),prec))
Работает со всеми приведенными примерами:
>>> rnd(500.99,0.1,0)
'500'
>>> rnd(500.99,0.1,0,'')
'501'
>>> rnd(100.39,0.2,1,'')
'100.4'
>>> rnd(100.39,0.2,1)
'100.2'
>>> rnd(100.59,0.2,1,'')
'100.6'
>>> rnd(100.59,0.2,1)
'100.4'
>>> rnd(5.019,0.01,2)
'5.01'
>>> rnd(5.019,0.01,2,'')
'5.02'
(Вызвать функцию с аргументом ключевого слова direction=
с любым строковым значением, отличным от 'down'
, чтобы подняться. Измените по своему вкусу ...)
Редактировать 2
Возвращает метод, ближайший к введенному номеру:
def rnd(n,f,prec):
n1=int(n) if prec==0 else round(n-math.fmod(n,f),prec)
n2=float(int(round(n))) if prec==0 else round((n+f)-math.fmod((n+f),f),prec)
return str(min((n1,n2), key=lambda e: math.fabs(e-n)))