2008-10-23 4 views
10

Есть ли какая-либо функция библиотеки для этой цели, поэтому я не делаю этого вручную, а риск заканчивается на TDWTF?Как округлить/потолковать/пол bcmath число в PHP?

echo ceil(31497230840470473074370324734723042.6); 

// Expected result 
31497230840470473074370324734723043 

// Prints 
<garbage> 
+0

Смотрите также мой связанный с этим вопрос: http://stackoverflow.com/questions/1642614/how-to-ceil-floor-and-round-bcmath-numbers – 2009-10-29 10:02:15

ответ

7

Это будет работать для вас:

$x = '31497230840470473074370324734723042.9'; 

bcscale(100); 
var_dump(bcFloor($x)); 
var_dump(bcCeil($x)); 
var_dump(bcRound($x)); 

function bcFloor($x) 
{ 
    $result = bcmul($x, '1', 0); 
    if ((bccomp($result, '0', 0) == -1) && bccomp($x, $result, 1)) 
     $result = bcsub($result, 1, 0); 

    return $result; 
} 

function bcCeil($x) 
{ 
    $floor = bcFloor($x); 
    return bcadd($floor, ceil(bcsub($x, $floor)), 0); 
} 

function bcRound($x) 
{ 
    $floor = bcFloor($x); 
    return bcadd($floor, round(bcsub($x, $floor)), 0); 
} 

В основном это находит flooy путем умножения на один с нулевой точностью.

Тогда он может сделать CEIL/раунд путем вычитания, что из общего числа, вызывая встроенные функции, а затем добавить результат обратно на

Edit: фиксированная для минуса чисел

+0

+1 , но, возможно, стоит добавить аргумент scale для bcCeil и bcRound, поскольку поведение зависит от масштаба. Если вы вызываете `bcscale (0)`, тогда попробуйте bcCeil ('1.1'), вы получите '1' не '2', как вы могли бы ожидать. Разрешение заданной шкалы будет соответствовать другим функциям BCMath. – 2014-10-28 01:54:15

+0

Также следует отметить, что аргумент scale должен иметь значение по умолчанию null и не должен перезаписывать значение, установленное `bcscale`, если оно не указано. – 2014-10-28 02:25:28

13

UPDATE: Смотрите мой улучшенный ответ здесь: How to ceil, floor and round bcmath numbers?.


Эти функции, кажется, больше смысла, по крайней мере, для меня:

function bcceil($number) 
{ 
    if ($number[0] != '-') 
    { 
     return bcadd($number, 1, 0); 
    } 

    return bcsub($number, 0, 0); 
} 

function bcfloor($number) 
{ 
    if ($number[0] != '-') 
    { 
     return bcadd($number, 0, 0); 
    } 

    return bcsub($number, 1, 0); 
} 

function bcround($number, $precision = 0) 
{ 
    if ($number[0] != '-') 
    { 
     return bcadd($number, '0.' . str_repeat('0', $precision) . '5', $precision); 
    } 

    return bcsub($number, '0.' . str_repeat('0', $precision) . '5', $precision); 
} 

Они поддерживают отрицательные числа и точности аргумент для функции bcround().

Некоторые тесты:

assert(bcceil('4.3') == ceil('4.3')); // true 
assert(bcceil('9.999') == ceil('9.999')); // true 
assert(bcceil('-3.14') == ceil('-3.14')); // true 

assert(bcfloor('4.3') == floor('4.3')); // true 
assert(bcfloor('9.999') == floor('9.999')); // true 
assert(bcfloor('-3.14') == floor('-3.14')); // true 

assert(bcround('3.4', 0) == number_format('3.4', 0)); // true 
assert(bcround('3.5', 0) == number_format('3.5', 0)); // true 
assert(bcround('3.6', 0) == number_format('3.6', 0)); // true 
assert(bcround('1.95583', 2) == number_format('1.95583', 2)); // true 
assert(bcround('5.045', 2) == number_format('5.045', 2)); // true 
assert(bcround('5.055', 2) == number_format('5.055', 2)); // true 
assert(bcround('9.999', 2) == number_format('9.999', 2)); // true