如何在PHP中执行正确的无符号右移


How do I perform a proper unsigned right shift in 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

有可能得到相同的结果吗?

最接近我想要的函数在这里:

无符号右移函数对负输入无效

试试这个函数:

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

最后我想我找到了解决方案,我不知道为什么,但这段代码为我工作:

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