您的位置:首页 > 编程语言 > PHP开发

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

2016-01-31 17:14 781 查看
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
.
"

"
;

echo
$ip_int
.
"

"
;

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
.
"

"
;

echo
$ip_int
.
"

"
;

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
.
"

"
;

echo
$ip_int
.
"

"
;

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
<</code>
count
[code](
$iparr
);
$i
++){

        
$num

+=
intval
(
$iparr
[
$i
]) *
pow(256,
count
(
$iparr
)-(
$i
+1));

    
}

    
return

$num
;

}

 
echo
 
$ip
.
'

'
;

$ip_int
=
ipToInt(
$ip
);

echo
$ip_int
.
'

'
;

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
.
'

'
;

$ip_int
=
ipToInt(
$ip
);

echo
$ip_int
.
'

'
;

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
'

'
;

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
'

'
;

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类型处理,要么自己写函数。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: