PHP函数ip2long()返回值为负数的解决办法

分享于:2017-09-26 15:06:24

PHP的ip2long 是将IP地址转换为数值的函数。

ip2long() 转出来的数值应该都是正整数,但是在某些机器转出负数, 刚开始以为是 PHP 版本问题, 后来做些测试, 确定是系统版本 32bits 和 64bits 的问题。

32 bits ip2long(): -2147483648 ~ 214748364764
64 bits ip2long(): 0 ~ 42949672945

测试代码:

1// 自己做转换, 这方法计算出来的数值是正确的.(32bits / 64bits 皆正确)
2function iptolong($ip){
3    list($a$b$c$d) = split("."$ip);
4    $ip_long = (($a * 256 + $b) * 256 + $c) * 256 + $d;
5    return $ip_long;
6}

另外做测试,在return值加上intval(), 如下述:

1// 这样子的数值就会产生负数.
2function iptolong($ip){
3    list($a$b$c$d) = split("."$ip);
4    $ip_long = (($a * 256 + $b) * 256 + $c) * 256 + $d;
5    return intval($ip_long);
6}

intval在32bits/64 bits最大值是不同的,于intval()里面有写到下述:

The maximum value depends on the system.
32 bit systems have a maximum signed integer range of -2147483648 to 2147483647.
So for example on such a system, intval("1000000000000") will return 2147483647.
The maximum signed integer value for 64 bit systems is 9223372036854775807.

另外做其它测试:

01//ip2long() 于 32bits 的系统测试
02ip2long("127.255.255.255"); // 2147483647 = 十进制的最大值
03ip2long("255.255.255.255"); // -1
04ip2long("255.255.255.254"); // -2
05ip2long("192.168.1.2"); // -1062731518
06ip2long() 于 64bits 的系统测试
07ip2long("127.255.255.255"); // 2147483647 = 十进制的最大值
08ip2long("255.255.255.255"); // 4294967295
09ip2long("255.255.255.254"); // 4294967294
10ip2long("192.168.1.2"); // 3232235778
11//知道问题是 32bits 系统造成的, 就很好解决囉~

解法1 - 自己转换

1function iptolong($ip){
2    list($a$b$c$d) = explode("."$ip);
3        return (($a * 256 + $b) * 256 + $c) * 256 + $d;
4}

解法2 - 转成二进制, 再转回十进制

1// bindec 只吃 string, 回传 double
2// decbin 会回传 string
3echo bindec(decbin(ip2long("192.168.1.2"))); // 3232235778

解法3 - 官方建议的解法 (推荐用此方法)

1// 直接印值, 使用 printf("%u")
2printf("%u"ip2long("192.168.1.2")); // 3232235778
3// 回传值(于 function 或 echo 等), 使用 sprintf("%u")
4echo sprintf("%u"ip2long("192.168.1.2")); // 3232235778

关于第一种方法,从数值转换回IP可以使用下面的参考代码(如果数特别大,在32bits下就不是整数,是浮点数,求余那个地方会出负数):

1function longtoip($num)  {
2    $d $num%256;
3    $c = (($num-$d)/256)%256;
4    $b = (($num-($c*256)-$d)/(256*256))%256;
5    $a = (($num-($b*256*256)-$c*256-$d)/(256*256*256))%256;
6    return $a.".".$b.".".$c.".".$d;  }


来源:http://blog.csdn.net/everything1209/article/details/22418559