PHP中IP地址与整型数字互相转换详解
IP转换成整型存储是数据库优化一大趋势,不少人目前存储IP时还在使用字符串类型存储,字符串索引比整型索引消耗资源很多,特别是表中数据量大的时候,以及求查询某一个ip段的数据,今天说的ip是指ip4,ip6不在本文范围内。
系统函数ip2long与long2ip
PHP中有内置函数ip2long可以将ip地址转换整型。
$ip='210.110.11.49'; echoip2long($ip);
输出:
-764540111
输出的整型有负号是因为我们得到的结果是有符号整型,有符号整型最大值2147483647,要把结果转换为无符号型可以这么写:
3530427185
使用long2ip把整型转换回ip地址
$ip='210.110.11.49'; $ip_int=ip2long($ip); echo$ip."<br/>"; echo$ip_int."<br/>"; echolong2ip($ip_int);
输出:
210.110.11.49 -764540111 210.110.11.49
从结果可以看到,ip与整型可以通过函数完成。
系统函数小bug
这中bug网上一搜都是,大意说的是ip某段加个前导0,先来看看这个bug实例
$ip='210.110.011.49'; $ip_int=ip2long($ip); echo$ip."<br/>"; echo$ip_int."<br/>"; echolong2ip($ip_int);
输出:
210.110.011.49 -764540623 210.110.9.49
转换结果不匹配,我们试着在ip第一段数字前加前导0,再看看
$ip='021.110.11.49'; $ip_int=ip2long($ip); echo$ip."<br/>"; echo$ip_int."<br/>"; echolong2ip($ip_int);
输出:
021.110.11.49 292424497 17.110.11.49
转换结果都出错。以上例子都是因为加了前导0后导致转换结果出错,连带逆转结果与原转换ip不匹配。
转换原理
目前有两个算法:
第一、第一段乘以256的三次方,第二段乘以256的平方,第三段乘以256、最后总和
$ip='0210.110.11.49'; functionipToInt($ip){ $iparr=explode('.',$ip); $num=0; for($i=0;$i<count($iparr);$i++){ $num+=intval($iparr[$i])*pow(256,count($iparr)-($i+1)); } return$num; } echo $ip.'<br/>'; $ip_int=ipToInt($ip); echo$ip_int.'<br/>'; echolong2ip($ip_int);
输出:
0210.110.11.49 3530427185 210.110.11.49
第二、通过位运算符
$ip='0210.110.11.49'; functionipToInt($ip){ $iparr=explode('.',$ip); return(intval($iparr[0]<<24))|(intval($iparr[1])<<16)|(intval($iparr[2])<<8)|(intval($iparr[3])); } echo $ip.'<br/>'; $ip_int=ipToInt($ip); echo$ip_int.'<br/>'; echolong2ip($ip_int);
输出:
0210.110.11.49 -764540111 210.110.11.49
检测IP是否合法
第一、自己遍历检测
functioncheck_ip($ip){ $iparr=explode('.',$ip); foreach($iparras$v){if($v>255)returnfalse;} returntrue; } echo'210.285.11.49,'; var_dump(check_ip('210.285.11.49')); echo'<br/>'; echo'210.205.11.49,'; var_dump(check_ip('210.205.11.49')); [code]
输出: [code] 210.285.11.49,bool(false) 210.205.11.49,bool(true)