PHP讀取純真IP數據庫的函數

Discuz  5.0 不在使用自己的IP數據,而是使用純真IP的數據格式, 存取純真IP數據庫稍微有點麻煩,它的存儲格式比較特殊也很有趣,具體的格式分析參考下面兩個鏈接,其他語言實現參考文章末的鏈接。

《純真IP數據庫格式詳解》
鏈接一:http://blog.csdn.Net/heiyeshuwu/archive/2006/05/12/725675.aspx
鏈接二:http://lumaqq.Linuxsir.org/article/qqwry_format_detail.html

純真IP數據庫官網:http://www.cz88.Net/ip/
純真IP數據庫下載:http://update.cz88.Net/soft/qqwry.rar

 

以下函數conrvertip()位於 Discuz!5_GBK/upload/include/misc.func.Php 路徑中,有興趣可以具體去閱讀分析。(下面代碼我做了簡單的修改,更便於閱讀,核心沒有修改)

 

<?
//===================================
//
// 功能:IP地址獲取真實地址函數
// 參數:$ip - IP地址
// 作者:[Discuz!] (C) Comsenz Inc.
//
//===================================

function convertip($ip) {
    
//IP數據文件路徑
    $dat_path = 'QQWry.Dat';

    
//檢查IP地址
    if(!preg_match("/^d{1,3}.d{1,3}.d{1,3}.d{1,3}$/", $ip)) {
        
return 'IP Address Error';
    }
    
//打開IP數據文件
    if(!$fd = @fopen($dat_path, 'rb')){
        
return 'IP date file not exists or access denied';
    }

    
//分解IP進行運算,得出整形數
    $ip = explode('.', $ip);
    
$ipNum = $ip[0* 16777216 + $ip[1* 65536 + $ip[2* 256 + $ip[3];

    
//獲取IP數據索引開始和結束位置
    $DataBegin = fread($fd, 4);
    
$DataEnd = fread($fd, 4);
    
$ipbegin = implode('', unpack('L', $DataBegin));
    
if($ipbegin < 0$ipbegin += pow(2, 32);
    
$ipend = implode('', unpack('L', $DataEnd));
    
if($ipend < 0$ipend += pow(2, 32);
    
$ipAllNum = ($ipend - $ipbegin/ 7 + 1;
    
    
$BeginNum = 0;
    
$EndNum = $ipAllNum;

    
//使用二分查找法從索引記錄中搜索匹配的IP記錄
    while($ip1num>$ipNum || $ip2num<$ipNum) {
        
$Middle= intval(($EndNum + $BeginNum/ 2);

        
//偏移指針到索引位置讀取4個字節
        fseek($fd, $ipbegin + 7 * $Middle);
        
$ipData1 = fread($fd, 4);
        
if(strlen($ipData1< 4) {
            
fclose($fd);
            
return 'System Error';
        }
        
//提取出來的數據轉換成長整形,如果數據是負數則加上2的32次冪
        $ip1num = implode('', unpack('L', $ipData1));
        
if($ip1num < 0$ip1num += pow(2, 32);
        
        
//提取的長整型數大於我們IP地址則修改結束位置進行下一次循環
        if($ip1num > $ipNum) {
            
$EndNum = $Middle;
            
continue;
        }
        
        
//取完上一個索引後取下一個索引
        $DataSeek = fread($fd, 3);
        
if(strlen($DataSeek< 3) {
            
fclose($fd);
            
return 'System Error';
        }
        
$DataSeek = implode('', unpack('L', $DataSeek.chr(0)));
        
fseek($fd, $DataSeek);
        
$ipData2 = fread($fd, 4);
        
if(strlen($ipData2< 4) {
            
fclose($fd);
            
return 'System Error';
        }
        
$ip2num = implode('', unpack('L', $ipData2));
        
if($ip2num < 0$ip2num += pow(2, 32);

        
//沒找到提示未知
        if($ip2num < $ipNum) {
            
if($Middle == $BeginNum) {
                
fclose($fd);
                
return 'Unknown';
            }
            
$BeginNum = $Middle;
        }
    }

    
//下面的代碼讀暈了,沒讀明白,有興趣的慢慢讀
    $ipFlag = fread($fd, 1);
    
if($ipFlag == chr(1)) {
        
$ipSeek = fread($fd, 3);
        
if(strlen($ipSeek< 3) {
            
fclose($fd);
            
return 'System Error';
        }
        
$ipSeek = implode('', unpack('L', $ipSeek.chr(0)));
        
fseek($fd, $ipSeek);
        
$ipFlag = fread($fd, 1);
    }

    
if($ipFlag == chr(2)) {
        
$AddrSeek = fread($fd, 3);
        
if(strlen($AddrSeek< 3) {
            
fclose($fd);
            
return 'System Error';
        }
        
$ipFlag = fread($fd, 1);
        
if($ipFlag == chr(2)) {
            
$AddrSeek2 = fread($fd, 3);
            
if(strlen($AddrSeek2< 3) {
                
fclose($fd);
                
return 'System Error';
            }
            
$AddrSeek2 = implode('', unpack('L', $AddrSeek2.chr(0)));
            
fseek($fd, $AddrSeek2);
        } 
else {
            
fseek($fd, -1, SEEK_CUR);
        }

        
while(($char = fread($fd, 1)) != chr(0))
            
$ipAddr2 .= $char;

        
$AddrSeek = implode('', unpack('L', $AddrSeek.chr(0)));
        
fseek($fd, $AddrSeek);

        
while(($char = fread($fd, 1)) != chr(0))
            
$ipAddr1 .= $char;
    } 
else {
        
fseek($fd, -1, SEEK_CUR);
        
while(($char = fread($fd, 1)) != chr(0))
            
$ipAddr1 .= $char;

        
$ipFlag = fread($fd, 1);
        
if($ipFlag == chr(2)) {
            
$AddrSeek2 = fread($fd, 3);
            
if(strlen($AddrSeek2< 3) {
                
fclose($fd);
                
return 'System Error';
            }
            
$AddrSeek2 = implode('', unpack('L', $AddrSeek2.chr(0)));
            
fseek($fd, $AddrSeek2);
        } 
else {
            
fseek($fd, -1, SEEK_CUR);
        }
        
while(($char = fread($fd, 1)) != chr(0)){
            
$ipAddr2 .= $char;
        }
    }
    
fclose($fd);

    
//最後做相應的替換操作後返回結果
    if(preg_match('/http/i', $ipAddr2)) {
        
$ipAddr2 = '';
    }
    
$ipaddr = "$ipAddr1 $ipAddr2";
    
$ipaddr = preg_replace('/CZ88.Net/is', '', $ipaddr);
    
$ipaddr = preg_replace('/^s*/is', '', $ipaddr);
    
$ipaddr = preg_replace('/s*$/is', '', $ipaddr);
    
if(preg_match('/http/i', $ipaddr|| $ipaddr == '') {
        
$ipaddr = 'Unknown';
    }

    
return $ipaddr;
}


//========================
//
//  調用舉例(速度很快)
//
//========================


echo convertip('219.238.235.10');
//輸出: 北京市 電信通

echo convertip('23.56.82.12');
//輸出:IANA

echo convertip('250.69.52.0');
//輸出:IANA保留地址

echo convertip('238.69.52.0');
//輸出:IANA保留地址 用於多點傳送

echo convertip('192.168.0.1');
//輸出:局域網 對方和您在同一內部網

echo convertip('255.255.255.255');
//輸出:純真網絡 2006年11月20日IP數據

?>



附:(相應其他實現程序)

Php)" href="http://www.coolcode.cn/?p=16" rel=bookmark>利用 QQWry.Dat 實現 IP 地址高效檢索(Php)(作者: andot)

數據庫(QQWry.Dat)查詢 C源碼" href="http://www.douzi.org/wp/index.Php/articles/71" rel=bookmark>純真IP數據庫(QQWry.Dat)查詢 C源碼 (作者:Windix)

 
發佈了56 篇原創文章 · 獲贊 0 · 訪問量 19萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章