這篇文章主要介紹了PHP中IP地址與整型數字互相轉換詳解,本文介紹了使用PHP函數ip2long與long2ip的使用,以及它們的BUG介紹,最後給出本身寫的兩個算法,須要的朋友能夠參考下mysql
IP轉換成整型存儲是數據庫優化一大趨勢,很多人目前存儲IP時還在使用字符串類型存儲,字符串索引比整型索引消耗資源不少,特別是表中數據量大的時候,以及求查詢某一個ip段的數據,今天說的ip是指ip4,ip6不在本文範圍內。算法
系統函數ip2long與long2ip
PHP中有內置函數ip2long能夠將ip地址轉換整型。sql
代碼以下:數據庫
$ip = '210.110.11.49'; echo ip2long($ip);
輸出:函數
-764540111
輸出的整型有負號是由於咱們獲得的結果是有符號整型,有符號整型最大值2147483647,要把結果轉換爲無符號型能夠這麼寫優化
3530427185
使用long2ip把整型轉換回ip地址spa
$ip = '210.110.11.49'; $ip_int = ip2long($ip); echo $ip."<br />"; echo $ip_int."<br />"; echo long2ip($ip_int);
輸出:code
210.110.011.49 -764540623 210.110.9.49
轉換結果不匹配,咱們試着在ip第一段數字前加前導0,再看看blog
$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不匹配。
轉換原理
目前有兩個算法:
第1、第一段乘以256的三次方,第二段乘以256的平方,第三段乘以25六、最後總和
$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
第2、經過位運算符
$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是否合法
第1、本身遍歷檢測
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)
第2、使用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類型處理,要麼本身寫函數。