浅谈PHP中IP与整型互相转换
2016-01-31 17:14
781 查看
IP转换成整型存储是数据库优化一大趋势,不少人目前存储IP时还在使用字符串类型存储,字符串索引比整型索引消耗资源很多,特别是表中数据量大的时候,以及求查询某一个ip段的数据,今天说的ip是指ip4,ip6不在本文范围内。
输出:
-764540111
输出的整型有负号是因为我们得到的结果是有符号整型,有符号整型最大值2147483647,要把结果转换为无符号型可以这么写
3530427185
使用long2ip把整型转换回ip地址
输出:
210.110.11.49
-764540111
210.110.11.49
从结果可以看到,ip与整型可以通过函数完成。
输出:
210.110.011.49
-764540623
210.110.9.49
转换结果不匹配,我们试着在ip第一段数字前加前导0,再看看
输出:
021.110.11.49
292424497
17.110.11.49
转换结果都出错。以上例子都是因为加了前导0后导致转换结果出错,连带逆转结果与原转换ip不匹配
第一、第一段乘以256的三次方,第二段乘以256的平方,第三段乘以256、最后总和
输出:
0210.110.11.49
3530427185
210.110.11.49
第二、通过位运算符
输出:
0210.110.11.49
-764540111
210.110.11.49
输出:
210.285.11.49,bool(false)
210.205.11.49,bool(true)
第二、使用ip2long返回
输出:
210.285.11.49,bool(false)
210.205.11.49,bool(true)
如果是mysql可以使用mysql系统函数INET_ATON与INET_NTOA解决,或者使用bigint类型处理,要么自己写函数。
系统函数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类型处理,要么自己写函数。
相关文章推荐
- Zend Studio12安装及破解
- PHP substr截取中文字符出现…
- php数组函数总结
- PHP 获取图像信息 getimagesize函数
- php 四种排序算法
- PHP-PDO笔记
- ThinkPHP - CURD增删改查 - 实例 - 搜索功能
- php常用的正则匹配
- thinkphp控制器中获取当前控制器路径
- ThinkPHP - 查询语句
- PHP学习注意点
- Thinkphp里import的几个使用方法介绍
- PHP开发学习门户改版效果图投票
- 如何使用php session
- 详细讲解PHP中缓存技术的应用
- ZendStudio快捷键
- PHP文件处理系统
- ThinkPHP单字母函数(快捷方法)使用总结
- yii2 日志功能使用记录
- PHP 三元运算符