浅谈PHP中IP与整型互相转换
来源:互联网 发布:说说大全软件 编辑:程序博客网 时间:2024/06/15 21:41
IP转换成整型存储是数据库优化一大趋势,不少人目前存储IP时还在使用字符串类型存储,字符串索引比整型索引消耗资源很多,特别是表中数据量大的时候,以及求查询某一个ip段的数据,今天说的ip是指ip4,ip6不在本文范围内。
系统函数ip2long与long2ip
PHP中有内置函数ip2long可以将ip地址转换整型。
$ip
=
'210.110.11.49'
;
echo
ip2long
(
$ip
);
输出:
-764540111
输出的整型有负号是因为我们得到的结果是有符号整型,有符号整型最大值2147483647,要把结果转换为无符号型可以这么写
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
从结果可以看到,ip与整型可以通过函数完成。
系统函数小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
转换结果都出错。以上例子都是因为加了前导0后导致转换结果出错,连带逆转结果与原转换ip不匹配
转换原理
目前有两个算法:
第一、第一段乘以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'
));
输出:
210.285.11.49,bool(false)
210.205.11.49,bool(true)
第二、使用ip2long返回
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类型处理,要么自己写函数。
- 浅谈PHP中IP与整型互相转换
- 浅谈PHP中IP与整型互相转换
- 嵌入式 整型ip与点式ip互相转换
- IP与数字互相转换(PHP )
- php的IP转换成整型函数
- ip地址与整数互相转换
- Java中整型与字符串的互相转化
- IP转换为整型
- PHP数组与XML互相转换
- MFC中整型与字符串转换
- 字符型IP地址与整型之间转换
- Oracle IP地址整型与字符串型相互转换
- IP地址与整型数之间的转换
- 关于IP地址与整型之间的转换
- php 字符串整型转换
- JAVA 整型和字符串互相转换
- MFC整型和字符串互相转换
- Cocos2d中CCSprite与UIImage互相转换
- 有线无线一体化网络设计原则
- 提高效率的Android Studio插件
- Hive复杂查询时,多层union可以转换为left join,左侧临时维度表生成方式
- 网桥、路由器以及网关三者的区别
- Hive,Pig区别
- 浅谈PHP中IP与整型互相转换
- int main(int argc,char* argv[])详解
- [LeetCode]Maximum Depth of Binary Tree
- c++中的虚函数和纯虚函数
- cocos2dx 3.x打包出现Can't find config file .cocos-project.json in path
- 解决DrawerLayout以及viewpager点击或者滑动后卡顿的问题
- ZJOI 树的统计 [五星]
- MFC中光标从窗口1跳转到窗口2中的某个控件中
- Linux下通过mysqldump备份MySQL数据库成sql文件