浅谈PHP中IP与整型互相转换

IP转换成整型存储是数据库优化一大趋势,不少人目前存储IP时还在使用字符串类型存储,字符串索引比整型索引消耗资源很多,特别是表中数据量大的时候,以及求查询某一个ip段的数据,今天说的ip是指ip4,ip6不在本文范围内。

系统函数ip2long与long2ip

PHP中有内置函数ip2long可以将ip地址转换整型。

1
2
$ip = '210.110.11.49' ;
echo ip2long ( $ip );

输出:

-764540111

输出的整型有负号是因为我们得到的结果是有符号整型,有符号整型最大值2147483647,要把结果转换为无符号型可以这么写

3530427185

使用long2ip把整型转换回ip地址

1
2
3
4
5
$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

从结果可以看到,ip与整型可以通过函数完成。

系统函数小bug

这中bug网上一搜都是,大意说的是ip某段加个前导0,先来看看这个bug实例

1
2
3
4
5
$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,再看看

1
2
3
4
5
$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

转换结果都出错。以上例子都是因为加了前导0后导致转换结果出错,连带逆转结果与原转换ip不匹配

转换原理

目前有两个算法:

第一、第一段乘以256的三次方,第二段乘以256的平方,第三段乘以256、最后总和

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
$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

第二、通过位运算符

1
2
3
4
5
6
7
8
9
10
11
$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是否合法

第一、自己遍历检测

1
2
3
4
5
6
7
8
9
10
11
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' ));

输出:

210.285.11.49,bool(false)
210.205.11.49,bool(true)

第二、使用ip2long返回

1
2
3
4
5
6
7
8
9
10
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类型处理,要么自己写函数。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值