PHP по модулю против substract в PHP_INT_MAX
В какой-то момент у меня был этот блок кода:
while( $i> $l-1 )
{
$x= fmod($i,$l);
$i= floor($i/$l);
}
Я решил избавиться от операции по модулю и написал этот блок:
while( true )
{
$d= floor( $i/$l );
if( $d>= 1 )
{
$x= $i - ($d*$l);
$i= $d;
}
else
{
break;
}
}
$ X используется для индексации массива длины $ l. $ I здесь под вопросом.
В то время как для некоторого относительно небольшого начального $ i оба блока дают одинаковый $x на всех итерациях, но при инициализации с чем-то близким к PHP_INT_MAX два блока не дают одинаковый $x.
К сожалению, $ l не может стать степенью 2, чтобы использовать битовые операторы, поэтому я застрял с этим.
Я предполагаю, что это как-то связано с происходящими внутренними закруглениями. Может ли fmod быть так оптимизирован для этого случая? Есть что-то, чего я не вижу?
Дополнительный комментарий после принятия ответа @trincot.
Одна вещь, которую я должен был упомянуть, состоит в том, что хотя можно ожидать, что второй метод даст лучшие результаты, из-за использования простого вычитания, это не так. Возможно, из-за деления в начале цикла (вот почему я спросил: "Может быть, fmod так оптимизирован)?
1 ответ
Согласно документации, fmod
работает на поплавках:
fmod - возвращает остаток с плавающей запятой (по модулю) от деления аргументов
Вместо этого оператор по модулю (%
) подойдет больше для того что нужно
Операнды модуля перед обработкой преобразуются в целые числа (путем удаления десятичной части).
fmod
станет неточным для больших целых чисел, поскольку представление с плавающей запятой не имеет такой же точности.
Примеры некоторых странностей, которые случаются:
$l=3;
$i=9223372036854775295;
echo is_int($i) . "<br>"; // 1 (true)
echo (9223372036854775295==$i) . "<br>"; // 1 (true)
echo number_format($i, 0, ".", "") . "<br>"; // 9223372036854774784
echo fmod($i,$l) . "<br>"; // 1
echo fmod($i-1,$l) . "<br>"; // 1
echo fmod($i-2,$l) . "<br>"; // 1
echo ($i % $l) . "<br>"; // 2
echo (($i-1) % $l) . "<br>"; // 1
echo (($i-2) % $l) . "<br>"; // 0
Обратите внимание, как просто number_format
уже разрушает точность целого числа; он возвращает другое число из-за преобразования с плавающей запятой.
Обратите также внимание, что это отсутствие точности делает fmod
верните 1 для трех последовательных чисел, в то время как оператор по модулю делает то, что вы хотите.
Таким образом, вы, кажется, намного лучше с %
,
альтернатива
Кажется, ваша функция разбивает число на "цифры" в L-образной форме. Например, когда $l=2
, ваш $x
-последовательность производит двоичное представление числа, за исключением последней цифры, которую вы пропустите.
В этом отношении вы могли бы взглянуть на вызов функции base_convert($i,10,$l)
, который производит одну цифру, соответствующую значению $x
в вашем коде, с буквами для цифр выше 9. Функция может принимать $l
значения до 36.