web-dev-qa-db-fra.com

Arrondir au multiple de cinq le plus proche PHP

Je veux une fonction php qui retourne 55 quand on l'appelle avec 52.

J'ai essayé la fonction round()

echo round(94, -1); // 90

Il retourne 90 mais je veux 95 .

Merci.

53
ahmadbrkat

Cela peut être accompli de différentes manières, selon votre convention d'arrondi préférée:

1. Arrondissez au suivant multiple de 5, excluez le nombre actuel

Comportement: 50 sorties 55, 52 sorties 55

function roundUpToAny($n,$x=5) {
    return round(($n+$x/2)/$x)*$x;
}

2. Arrondir au le plus proche multiple de 5, inclure le nombre actuel

Comportement: 50 sorties 50, 52 sorties 55, 50,25 sorties 50  

function roundUpToAny($n,$x=5) {
    return (round($n)%$x === 0) ? round($n) : round(($n+$x/2)/$x)*$x;
}

3. Arrondir à un entier, puis au multiple de 5 le plus proche

Comportement: 50 sorties 50, 52 sorties 55, 50,25 sorties 55

function roundUpToAny($n,$x=5) {
    return (ceil($n)%$x === 0) ? ceil($n) : round(($n+$x/2)/$x)*$x;
}
124
SW4
  1. Diviser par 5
  2. round() (ou ceil()si vous voulez arrondir toujours)
  3. Multipliez par 5.

La valeur 5 (la résolution/granularité) peut être n'importe quoi - l'a remplacée aux étapes 1 et 3

64
jensgram

Arrondir vers le bas:

$x = floor($x/5) * 5;

Rassembler:

$x = ceil($x/5) * 5;

Arrondir au plus proche (haut ou bas):

$x = round($x/5) * 5;
52
PaulJWilliams

Essayez cette petite fonction que j'ai écrite.

function ceilFive($number) {
    $div = floor($number / 5);
    $mod = $number % 5;

    If ($mod > 0) $add = 5;
    Else $add = 0;

    return $div * 5 + $add;
}

echo ceilFive(52);
3
Knight
   echo $value - ($value % 5);

Je sais que c’est une vieille question, mais à mon humble avis, l’utilisation de l’opérateur de module est la meilleure solution, et bien plus élégante que la réponse acceptée.

3
Mahn

From Gears bibliothèque

MathType::roundStep(50, 5); // 50
MathType::roundStep(52, 5); // 50
MathType::roundStep(53, 5); // 55

MathType::floorStep(50, 5); // 50
MathType::floorStep(52, 5); // 50
MathType::floorStep(53, 5); // 50

MathType::ceilStep(50, 5); // 50
MathType::ceilStep(52, 5); // 55
MathType::ceilStep(53, 5); // 55

La source:

public static function roundStep($value, int $step = 1)
{
    return round($value / $step) * $step;
}

public static function floorStep($value, int $step = 1)
{
    return floor($value / $step) * $step;
}

public static function ceilStep($value, int $step = 1)
{
    return ceil($value / $step) * $step;
}
2
Cosmologist

Je le fais comme ça:

private function roundUpToAny(int $n, $x = 9)
{
    return (floor($n / 10) * 10) + $x;
}

Tests:

assert($this->roundUpToAny(0, 9) == 9);
assert($this->roundUpToAny(1, 9) == 9);
assert($this->roundUpToAny(2, 9) == 9);
assert($this->roundUpToAny(3, 9) == 9);
assert($this->roundUpToAny(4, 9) == 9);
assert($this->roundUpToAny(5, 9) == 9);
assert($this->roundUpToAny(6, 9) == 9);
assert($this->roundUpToAny(7, 9) == 9);
assert($this->roundUpToAny(8, 9) == 9);
assert($this->roundUpToAny(9, 9) == 9);
assert($this->roundUpToAny(10, 9) == 19);
assert($this->roundUpToAny(11, 9) == 19);
assert($this->roundUpToAny(12, 9) == 19);
assert($this->roundUpToAny(13, 9) == 19);
assert($this->roundUpToAny(14, 9) == 19);
assert($this->roundUpToAny(15, 9) == 19);
assert($this->roundUpToAny(16, 9) == 19);
assert($this->roundUpToAny(17, 9) == 19);
assert($this->roundUpToAny(18, 9) == 19);
assert($this->roundUpToAny(19, 9) == 19);
1
Vaci

Voici ma version de Musthafa's function. Celui-ci est plus complexe mais supporte pour les nombres flottants ainsi que pour les entiers. Le nombre à arrondir peut également être une chaîne.

/**
 * @desc This function will round up a number to the nearest rounding number specified.
 * @param $n (Integer || Float) Required -> The original number. Ex. $n = 5.7;
 * @param $x (Integer) Optional -> The nearest number to round up to. The default value is 5. Ex. $x = 3;
 * @return (Integer) The original number rounded up to the nearest rounding number.
 */
function rounduptoany ($n, $x = 5) {

  //If the original number is an integer and is a multiple of 
  //the "nearest rounding number", return it without change.
  if ((intval($n) == $n) && (!is_float(intval($n) / $x))) {

    return intval($n);
  }
  //If the original number is a float or if this integer is 
  //not a multiple of the "nearest rounding number", do the 
  //rounding up.
  else {

    return round(($n + $x / 2) / $x) * $x;
  }
}

J'ai essayé les fonctions de Knight , Musthafa et même la suggestion de Praesagus . Ils ne prennent pas en charge les nombres flottants et les solutions de Musthafa's & Praesagus ne fonctionnent pas correctement avec certains chiffres. Essayez les numéros de test suivants et faites la comparaison vous-même:

$x= 5;

$n= 200;       // D = 200     K = 200     M = 200     P = 205
$n= 205;       // D = 205     K = 205     M = 205     P = 210
$n= 200.50;    // D = 205     K = 200     M = 200.5   P = 205.5
$n= '210.50';  // D = 215     K = 210     M = 210.5   P = 215.5
$n= 201;       // D = 205     K = 205     M = 200     P = 205
$n= 202;       // D = 205     K = 205     M = 200     P = 205
$n= 203;       // D = 205     K = 205     M = 205     P = 205

** D = DrupalFever K = Knight M = Musthafa P = Praesagus
1
DrupalFever

Multiplier par 2, arrondir à -1, diviser par 2.

Probablement, vous pouvez aussi envisager ce type de doublure. C'est plus rapide! Fonctionne pour $num >= 0 et $factor > 0.

$num = 52;
$factor = 55;
$roundedNum = $num + $factor - 1 - ($num + $factor - 1) % $factor;
0

Je viens d’écrire cette fonction en 20 minutes, à partir des nombreux résultats que j’ai trouvés ici et là, je ne sais pas pourquoi ça marche ni comment ça marche !! :RÉ

La conversion des numéros de devise de 151431.1 LBP en 150000.0 LBP m'intéressait principalement. (151431.1 LBP == ~ 100 USD) qui fonctionne parfaitement jusqu'à présent, mais j'ai essayé de le rendre compatible avec d'autres devises et numéros, mais je ne suis pas sûr que cela fonctionne correctement !!

/**
 * Example:
 * Input = 151431.1 >> return = 150000.0
 * Input = 17204.13 >> return = 17000.0
 * Input = 2358.533 >> return = 2350.0
 * Input = 129.2421 >> return = 125.0
 * Input = 12.16434 >> return = 10.0
 *
 * @param     $value
 * @param int $modBase
 *
 * @return  float
 */
private function currenciesBeautifier($value, int $modBase = 5)
{
    // round the value to the nearest
    $roundedValue = round($value);

    // count the number of digits before the dot
    $count = strlen((int)str_replace('.', '', $roundedValue));

    // remove 3 to get how many zeros to add the mod base
    $numberOfZeros = $count - 3;

    // add the zeros to the mod base
    $mod = str_pad($modBase, $numberOfZeros + 1, '0', STR_PAD_RIGHT);

    // do the magic
    return $roundedValue - ($roundedValue % $mod);
}

N'hésitez pas à le modifier et à le réparer s'il y a un problème

0
Mahmoud Zalt
function round_up($n, $x = 5) {
  $rem = $n % $x;
  if ($rem < 3)
     return $n - $rem;
  else
     return $n - $rem + $x;
}
0
Musthafa