PHP100 2019-03-28
IP转换成整型存储是数据库优化一大趋势,不少人目前存储IP时还在使用字符串类型存储,字符串索引比整型索引消耗资源很多,特别是表中数据量大的时候,以及求查询某一个ip段的数据,今天说的ip是指ip4,ip6不在本文范围内。
系统函数ip2long与long2ip
PHP中有内置函数ip2long可以将ip地址转换整型。
代码如下:
$ip = '210.110.11.49'; echo ip2long($ip);
代码如下:
-764540111
代码如下:
3530427185
使用long2ip把整型转换回ip地址
代码如下:
$ip = '210.110.11.49'; $ip_int = ip2long($ip); echo $ip."<br />"; echo $ip_int."<br />"; echo long2ip($ip_int);
代码如下:
210.110.11.49 -764540111 210.110.11.49
系统函数小bug
这中bug网上一搜都是,大意说的是ip某段加个前导0,先来看看这个bug实例
代码如下:
$ip = '210.110.011.49'; $ip_int = ip2long($ip); echo $ip."<br />"; echo $ip_int."<br />"; echo long2ip($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 />"; echo long2ip($ip_int);
代码如下:
021.110.11.49 292424497 17.110.11.49
转换原理
目前有两个算法:
第一、第一段乘以256的三次方,第二段乘以256的平方,第三段乘以256、最后总和
代码如下:
$ip = '0210.110.11.49';
 
function ipToInt($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 />';
echo long2ip($ip_int);代码如下:
0210.110.11.49 3530427185 210.110.11.49
第二、通过位运算符
代码如下:
$ip = '0210.110.11.49';
 
function ipToInt($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 />';
echo long2ip($ip_int);代码如下:
0210.110.11.49 -764540111 210.110.11.49
检测IP是否合法
第一、自己遍历检测
代码如下:
function check_ip($ip){
    $iparr = explode('.',$ip);
    foreach($iparr as $v){ if($v>255) return false; }
    return true;
}
 
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]
<p>输出:
[code]
210.285.11.49,bool(false)
210.205.11.49,bool(true)
</p>代码如下:
function check_ip($ip){
    if(ip2long($ip)) return true;
    return false;
}
 
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'));代码如下:
210.285.11.49,bool(false) 210.205.11.49,bool(true)
后记
不少人把ip写库用ip2long转换存放int类型的字段中,但是,在不同的系统平台上,ip2long函数得到的值是不同的,因此可能造成在从数据库中读出数据逆转ip时用long2ip得到的ip与原ip不符合
如果是mysql可以使用mysql系统函数INET_ATON与INET_NTOA解决,或者使用bigint类型处理,要么自己写函数。