Как выполнить правильный беззнаковый сдвиг вправо в PHP?

Возможно ли получить одинаковые результаты в PHP и Javascript?

Пример:

JavaScript

<script>

function urshift(a, b)
{
  return a >>> b;
}

document.write(urshift(10,3)+"<br />");
document.write(urshift(-10,3)+"<br />");
document.write(urshift(33, 33)+"<br />");
document.write(urshift(-10, -30)+"<br />");
document.write(urshift(-14, 5)+"<br />");

</script>

выход:

1
536870910
16
1073741821
134217727

PHP

function uRShift($a, $b) 
    { 
        if ($a < 0) 
        { 
            $a = ($a >> 1); 
            $a &= 2147483647; 
            $a |= 0x40000000; 
            $a = ($a >> ($b - 1)); 
        } else { 
            $a = ($a >> $b); 
        } 
        return $a; 
    }

echo uRShift(10,3)."<br />";
echo uRShift(-10,3)."<br />";
echo uRShift(33,33)."<br />";
echo uRShift(-10,-30)."<br />";
echo uRShift(-14,5)."<br />";

выход:

1
536870910
0
0
134217727

Возможно ли получить такие же результаты?

Ближайшая функция к тому, что я хочу, находится здесь:

Функция сдвига вправо без знака не работает для отрицательного ввода

Спасибо за помощь.


person Kamil Wiśniewski    schedule 23.08.2014    source источник
comment
Возможный дубликат Unsigned Right Shift/Zero -fill Right Shift в PHP (эквивалент Java/JavaScript)   -  person frzsombor    schedule 21.06.2018


Ответы (2)


Попробуйте эту функцию:

function unsigned_shift_right($value, $steps) {
    if ($steps == 0) {
         return $value;
    }

    return ($value >> $steps) & ~(1 << (8 * PHP_INT_SIZE - 1) >> ($steps - 1));
}

Вывод на основе вашего примера кода:

1
536870910
16
1073741821
134217727
person silkfire    schedule 23.08.2014
comment
PHP дает мне результаты: 1 2305843009213693950 0 1073741823 576460752303423487 почему? - person Kamil Wiśniewski; 23.08.2014

Наконец, я думаю, что нашел решение, не знаю почему, но этот код работает для меня:

function uRShift($a, $b) 
    { 
        if ($b > 32 || $b < -32) {
            $m = (int)($b/32);
            $b = $b-($m*32);
        }

        if ($b < 0)
            $b = 32 + $b;

        if ($a < 0) 
        { 
            $a = ($a >> 1); 
            $a &= 2147483647; 
            $a |= 0x40000000; 
            $a = ($a >> ($b - 1)); 
        } else { 
            $a = ($a >> $b); 
        } 
        return $a; 
    }
person Kamil Wiśniewski    schedule 24.08.2014