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'));
[code]
輸出:
[code]
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類型處理,要麼自己寫函數。