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

PHP 2013-03-19 sprintf,ip2long,ip,long2ip

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与整型互相转换

文章地址:http://www.qttc.net/201303291.html

除非标注,琼台博客所有博文均为原创,转载请加文字链接注明来源

乳名?小名?昵称?网名?均可

email,放心,我不会给你乱投广告的

想获得回访就把你的站点URL写上(没有留空)

[NOTICE]木要投放广告
[NOTICE]木要骂人,说不该说的话
[NOTICE]自由言论,但要遵纪守法

Comments 2

  • 不需要吧。。。 `php -r "echo ip2long('210.110.11.49'),'<-->',long2ip('3530427185'),' ';"` ===output=== `3530427185<-->210.110.11.49 ` 截图: ![20170522_001_4b31.png](https://ooo.0o0.ooo/2017/05/22/592286479ce9e.png)
    2017-05-22 14:34:09 [ 跟帖 ]
    1 #
  • 第一次来访问您的博客,多多支持哦!~
    2013-03-20 09:36:08 [ 跟帖 ]
    2 #