2008-10-23 15 views
10

¿Hay alguna función de biblioteca para este propósito, entonces no lo hago a mano y me arriesgo a terminar en TDWTF?Cómo redondear/cementar/plantar un número bcmath en PHP?

echo ceil(31497230840470473074370324734723042.6); 

// Expected result 
31497230840470473074370324734723043 

// Prints 
<garbage> 
+0

Véase también mi pregunta relacionada: http://stackoverflow.com/questions/1642614/how-to-ceil-floor-and-round-bcmath-numbers –

Respuesta

7

Esto funcionará para usted:

$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); 
} 

Básicamente se encuentra el flooy multiplicando por uno con precisión cero.

entonces se puede hacer ceil/vuelta restando que del total, llamando el construido en funciones, a continuación, añadir el resultado de nuevo en

Editar: fijo para los números -ve

+0

+1 , pero podría valer la pena agregar un argumento de escala a bcCeil y bcRound, ya que el comportamiento depende de la escala. Si llama a 'bcscale (0)' y luego prueba bcCeil ('1.1'), obtendrá '1' no '2' como cabría esperar. Permitir que la escala sea especificada sería consistente con las otras funciones de BCMath. –

+0

También hay que tener en cuenta que el argumento de escala debe ser por defecto nulo y no debe sobrescribir el valor establecido por 'bcscale' si no se proporciona. –

13

ACTUALIZACIÓN: Véase mi respuesta mejorada aquí: How to ceil, floor and round bcmath numbers?.


Estas funciones parecen tener más sentido, al menos para mí:

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); 
} 

Apoyan los números negativos y el argumento de precisión para la función bcround().

Algunas pruebas:

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 
+0

No funcionará con enteros. Buena realización de estas funciones aquí: http://stackoverflow.com/a/1653826/541961 Alix, puede editar su publicación para vincularla a la más nueva. – Dmitriy

+1

@Dmitriy: Listo. Gracias. –

Cuestiones relacionadas