php讀取qqwry.dat ip地址數據庫文件程序

文章先來介紹關于qqwry.dat的文件內容結構然後根據它的特點我們可寫出讀取qqwry.dat ip庫的內容找到我們想要的內容。

首先看看QQWry.Data文件的內容結構 ,以及解讀方式 。

一、文件結構
文件主要分三個結構
1、文件頭,8個字節;
2、數據記錄區,不定長度;
3、索引區,長度爲 7 的整數倍;

二、文件頭
文件頭的8個字節分兩部分,每個部分4個字節,分別指定了索引區的開始地址和結束地址。所以可以通過兩個地址的差值 除 7 後 加 1 可以計算出總的記錄數。

二、記錄區
記錄區的數據需要通過索引區的數據來獲得各個數據的起始位置;本區數據記錄了IP地址的結束地址和地區字符串;所有地區字符串都以 0×00 爲結束。

三、索引區
檢索IP對應的地區,關鍵就是找到IP起始地址對應的索引內容。一個IP索引數據包含7個字節,前4個字節是IP地址起始值,後3個字節是對應的IP數據 記錄在文件內的偏移地址;IP數據記錄中,前 4 個字節是IP結束地址;緊跟的數據有兩種模式: 0×01 模式 和 0×02 模式。

0×01模式,即在IP數據的第5個字節是 0×01,則在後面的 3 個字節是國家地區數據的偏移地址;國家地區數據包括國家和地區這兩個字符串。即
—————————————————————
4字節 | 3字節 重定向 0x NN NN NN -> 國家地區數據的文件偏移地址
—————————————————————

0×02模式,即在IP數據的第5個字節是 0×02,則在後面的 3 個字節是國家數據的偏移地址,地區數據是再往後的字符串,以 0×00 截至。即
—————————————————————————–
4字節 | 3字節 重定向 0x NN NN NN -> 國家數據的文件偏移地址 | 地區字符串 | 0×00
—————————————————————————–

對于 0×01 模式所得到的 國家地區數據中,它可能又帶有一個重定向結構,即
————————————–
國家字符串 | 0×00 | 地區字符串 | 0×00
————————————–

————————————————————————-
國家字符串 | 0×00 | 0×02 | 3字節 0x NN NN NN -> 地區字符串的文件偏移地址
————————————————————————-

對于前一種情況,比較簡單,直接讀出兩個字符串數據就可以了;對于後一種情況,需要再次重定向到地區字符串的偏移地址,然後讀取到 0×00 爲字符串結尾。

對于這種采取地址映射實際字符串值的方式,主要作用是避免重複記錄字符串值。在整個IP地址庫文件中,有太多相同字符串記錄了,采用 3 字節的映射地址要比重複記錄字符串值節省太多空間了。

PHP代碼讀取操作QQWry.dat文件 :

代碼如下

function bin2ip($bin){
$ip = '';
$bd = str_split($bin, 1);
for($i = 4; $i > 0; $i--){
$ip .= "." . sprintf("%03d", implode('', unpack('s', $bd[$i-1] . chr(0))));
}
return substr($ip, 1);
}

//--------------------------------------------------
$f = fopen('QQWry.Dat', 'r');
$c = fread($f, 4);
$d = fread($f, 4);

$index_begin = implode('', unpack('L', $c));
$index_end = implode('', unpack('L', $d));
if($index_begin < 0) $index_begin += pow(2, 32);
if($index_end < 0) $index_end += pow(2, 32);

$ip_num = ($index_end - $index_begin) / 7 + 1;

echo "index begin at: $index_beginn";
echo "index end at: $index_endn";
echo "ip data count : $ip_numn";

$output = '';

for($i = 0; $i < $ip_num; $i++){

//文件指針指到每個IP數據文件的索引取得索引數據(7字節)上
fseek($f, $i * 7 + $index_begin);
$ip4 = fread($f, 4); //IP起始地址
if(strlen($ip4) < 4) exit('data file error');

$ip3 = fread($f, 3); //IP記錄偏移地址
if(strlen($ip3) < 3) exit('data file error');

$dataseek = implode('', unpack('L', $ip3 . chr(0)));
if($dataseek < 0) $index_ip_record += pow(2, 32);

//指向記錄區 $dataseek 位置查找記錄
fseek($f, $dataseek);
$ipdata = fread($f, 4); //IP結束地址
if(strlen($ipdata) < 4) exit('data file error');

$area = '';
$country = '';

//讀一個標記位
$flag = fread($f, 1);
if($flag == chr(1)){ //國家名偏移標記位 模式一 0x01
$area1seek = fread($f, 3);
if(strlen($area1seek) < 3) exit('data file error');
$area1seek = implode('', unpack('L', $area1seek . chr(0)));
fseek($f, $area1seek);
$flag = fread($f, 1); //可能又是標記位
}
if($flag == chr(2)){ //國家地區 重定向
$area1seek = fread($f, 3);
if(strlen($area1seek) < 3) exit('data file error');
$area1seek = implode('', unpack('L', $area1seek . chr(0)));
$flag = fread($f, 1);
if($flag == chr(2)){
$area2seek = fread($f, 3);
$area2seek = implode('', unpack('L', $area2seek . chr(0)));
fseek($f, $area2seek);
}else{
fseek($f, -1, SEEK_CUR);
}
while(($c = fread($f, 1)) != chr(0)) $area .= $c;
fseek($f, $area1seek);
while(($c = fread($f, 1)) != chr(0)) $country .= $c;
}else{
fseek($f, -1, SEEK_CUR);
while(($c = fread($f, 1)) != chr(0)) $country .= $c;

$flag = fread($f, 1); //如果地區是重定向的
if($flag == chr(2)){
$area2seek = fread($f, 3);
$area2seek = implode('', unpack('L', $area2seek . chr(0)));
fseek($f, $area2seek);
}else{
fseek($f, -1, SEEK_CUR);
}
while(($c = fread($f, 1)) != chr(0)) $area .= $c;
}
$adata = trim($country) . trim($area); //$country是國家字符串 , $area 是地區字符串
}
fclose($f);

這個函數我們看到最多的就是文件操作相關函數如fopen,fseek,fread這些,有需要的朋友可以看看。

更多相關文章
  • php操作xml類主要是針對xml文件夾一個讀取查詢刪除數據例子,下面一起來看看,目前此類暫只支持三級節點哦.文件名:xml.class.php 代碼如下 /*** * ** 文件名: xml.php ** 作 用: xml類,完善中,暫只支持三級節點 ** 作 者: loking(biyees)
  • 今天看到一個朋友分享兩處隨機讀取N條記錄的文章,下面我們做了一下之後會發現可以隨機做個更新了,下面一起來看看吧.隨機讀取數據MYSQL自身語句便可實現,下面舉例來說明:比如,我要隨機讀取6條數據顯示給浏覽者:MYSQ
  • 本文章來給各位同學介紹一個PHP tclip擴展調用opencv從記憶體中讀取及解析圖片數據的問題,希望文章能幫助到大家哦.前兩天蠻說有一個PHP擴展,tclip圖片裁剪程序.Tclip致力于人臉識別和圖片顯著性區域的識別,它與其他圖片裁剪程序最大的不同之處在于,在裁剪後的圖片中保留人臉和其他圖片顯 ...
  • 解決phpMyAdmin不能導入大于2M數據庫文件
    phpMyAdmin不能導入2MB以上的文件是因爲我們的php.ini配置作了限制了,如果有權限我們可以在php.ini配置中修改,如果沒有權限可以使用第三方工具進行備份,下面整理了一些方法.當我們使用phpMyAdmin導入備份的數據庫文件時,默認只能上傳小于2M的文件.但是通常我們的數據庫備份文 ...
  • php教程 fputcsv() 函數csv數據讀寫數據庫教程文件代碼fputcsv() 函數用于將數據格式爲csv格式,以便寫入文件或者數據庫.1.將字符串寫入csv文件中 $test_array = array( array("111","sdfsd",&q ...
  • 下文爲各位整理一個PHP的實現一個高效的數據庫(文件存儲,NOSQL)的例子,希望對各位有幫助. 用文件的方式讀寫,一個文件是索引文件,另一個文件是真實的數據文件.索引文件分爲2部分,第一部分是所有的指針,記錄第二部
  • mysql  LOAD DATA INFILE 命令模式導入超大數據庫文件
    很多我程序員導入數據一般都使用phpmyadmin或直接打包數據庫放在目錄了,而phpmy
  • 已備份數據庫教程的磁盤上結構版本爲611. 伺服器支持版本539, 無法還原或升級此數據庫,restore database 操作異常終止.遠程伺服器主機上安裝了sql server 2005,備份的時候使用虛擬主機自帶的還原工具備份數據庫爲data.dat格式的,在本機sql2000想還原數據庫的
一周排行
  • 努比亞z7mini調整螢幕字體大小方法圖
    努比亞z7mini調整螢幕字體大小方法非常的簡單我們只要簡單的在在手機的設置中進行簡單的操 ...
  • mongodb也是一種輕型的數據庫,下面我來介紹java與mongodb數據庫的應用,包括寫,讀取,保存mongodb數據庫的實例,各位朋友有需要可參考.下面的例子是把User對象插入到Mongodb中,並讀出來 代 ...
  • 在以前如果我們要實現php+ajax實現多域名跨域登錄的話很多朋友都碰到無法正常跨域登錄問題,下面我來給大家解決跨域名登錄問題,有需要的參考.該同步登陸需求需滿足以下三個關鍵點:1)A域名下登陸的用戶,跳轉到B域名下
  • 代碼如下 using System;using System.Data;using System.Data.Common;using System.Configuration;public class DbHelpe
  • SublimeLinter是前端編碼利器--Sublime Text 的一款插件,用于高亮提示用戶編寫的代碼中存在的不規範和錯誤的寫法,現在我們來解決一下Mac下SublimeLinter的Unsafe Charac
  • 下面一起來看一篇關于linux下用wget下載jdk教程,希望本例子對各位同學會帶來幫助的哦.oracle jdk的下載真是一件蛋疼的事情,本地adsl才幾kb/s.這要下載到猴年馬月啊.獲取下載地址用wget下載發 ...
  • 功能:split 函數可返回基于 0 的一維數組,其中包含指定數目的子字符串語法:split(expression[,delimiter[,count[,compare]]])參數:參數描述 expression必選 ...
  • 可能有很多朋友都會碰到一個問題不知道127.0.0.1與localhost有什麽區別,但是有的時間會發現使用localhost連接不了,但是改成127.0.0.1即可了連接了,那麽他們會有什麽區別呢,下面我給大家介紹 ...
  • wget是linux中遠程文件下載命令了我們在安裝軟件或工具時可以直接使用wget下載文件然後通過make進行安裝了,但有時使用wget時會提示wget: unable to resolve host address ...
  • 本文章介紹了關于最新 Ubuntu12.04 64bit環境安裝教程,有需要了解的朋友可以參考一下下.操作系統版本:Ubuntu 12.04 64bit使用root賬號登陸系統,建立相關的目錄下載存放路徑/home/ ...