Discuz 5.0 中读取纯真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 路径中,有兴趣可以具体去阅读分析。(下面代码我做了简单的修改,更便于阅读,核心没有修改)

  1. <?
  2. //===================================
  3. //
  4. // 功能:IP地址获取真实地址函数
  5. // 参数:$ip - IP地址
  6. // 作者:[Discuz!] (C) Comsenz Inc.
  7. //
  8. //===================================
  9. function convertip($ip) {
  10.     //IP数据文件路径
  11.     $dat_path = 'QQWry.Dat';
  12.  
  13.     //检查IP地址
  14.     if(!preg_match("/^d{1,3}.d{1,3}.d{1,3}.d{1,3}$/", $ip)) {
  15.         return 'IP Address Error';
  16.     }
  17.     //打开IP数据文件
  18.     if(!$fd = @fopen($dat_path, 'rb')){
  19.         return 'IP date file not exists or access denied';
  20.     }
  21.  
  22.     //分解IP进行运算,得出整形数
  23.     $ip = explode('.', $ip);
  24.     $ipNum = $ip[0] * 16777216 + $ip[1] * 65536 + $ip[2] * 256 + $ip[3];
  25.  
  26.     //获取IP数据索引开始和结束位置
  27.     $DataBegin = fread($fd, 4);
  28.     $DataEnd = fread($fd, 4);
  29.     $ipbegin = implode('', unpack('L', $DataBegin));
  30.     if($ipbegin < 0) $ipbegin += pow(2, 32);
  31.     $ipend = implode('', unpack('L', $DataEnd));
  32.     if($ipend < 0) $ipend += pow(2, 32);
  33.     $ipAllNum = ($ipend - $ipbegin) / 7 + 1;
  34.     
  35.     $BeginNum = 0;
  36.     $EndNum = $ipAllNum;
  37.  
  38.     //使用二分查找法从索引记录中搜索匹配的IP记录
  39.     while($ip1num>$ipNum || $ip2num<$ipNum) {
  40.         $Middle= intval(($EndNum + $BeginNum) / 2);
  41.  
  42.         //偏移指针到索引位置读取4个字节
  43.         fseek($fd, $ipbegin + 7 * $Middle);
  44.         $ipData1 = fread($fd, 4);
  45.         if(strlen($ipData1) < 4) {
  46.             fclose($fd);
  47.             return 'System Error';
  48.         }
  49.         //提取出来的数据转换成长整形,如果数据是负数则加上2的32次幂
  50.         $ip1num = implode('', unpack('L', $ipData1));
  51.         if($ip1num < 0) $ip1num += pow(2, 32);
  52.         
  53.         //提取的长整型数大于我们IP地址则修改结束位置进行下一次循环
  54.         if($ip1num > $ipNum) {
  55.             $EndNum = $Middle;
  56.             continue;
  57.         }
  58.         
  59.         //取完上一个索引后取下一个索引
  60.         $DataSeek = fread($fd, 3);
  61.         if(strlen($DataSeek) < 3) {
  62.             fclose($fd);
  63.             return 'System Error';
  64.         }
  65.         $DataSeek = implode('', unpack('L', $DataSeek.chr(0)));
  66.         fseek($fd, $DataSeek);
  67.         $ipData2 = fread($fd, 4);
  68.         if(strlen($ipData2) < 4) {
  69.             fclose($fd);
  70.             return 'System Error';
  71.         }
  72.         $ip2num = implode('', unpack('L', $ipData2));
  73.         if($ip2num < 0) $ip2num += pow(2, 32);
  74.  
  75.         //没找到提示未知
  76.         if($ip2num < $ipNum) {
  77.             if($Middle == $BeginNum) {
  78.                 fclose($fd);
  79.                 return 'Unknown';
  80.             }
  81.             $BeginNum = $Middle;
  82.         }
  83.     }
  84.  
  85.     //下面的代码读晕了,没读明白,有兴趣的慢慢读
  86.     $ipFlag = fread($fd, 1);
  87.     if($ipFlag == chr(1)) {
  88.         $ipSeek = fread($fd, 3);
  89.         if(strlen($ipSeek) < 3) {
  90.             fclose($fd);
  91.             return 'System Error';
  92.         }
  93.         $ipSeek = implode('', unpack('L', $ipSeek.chr(0)));
  94.         fseek($fd, $ipSeek);
  95.         $ipFlag = fread($fd, 1);
  96.     }
  97.  
  98.     if($ipFlag == chr(2)) {
  99.         $AddrSeek = fread($fd, 3);
  100.         if(strlen($AddrSeek) < 3) {
  101.             fclose($fd);
  102.             return 'System Error';
  103.         }
  104.         $ipFlag = fread($fd, 1);
  105.         if($ipFlag == chr(2)) {
  106.             $AddrSeek2 = fread($fd, 3);
  107.             if(strlen($AddrSeek2) < 3) {
  108.                 fclose($fd);
  109.                 return 'System Error';
  110.             }
  111.             $AddrSeek2 = implode('', unpack('L', $AddrSeek2.chr(0)));
  112.             fseek($fd, $AddrSeek2);
  113.         } else {
  114.             fseek($fd, -1, SEEK_CUR);
  115.         }
  116.  
  117.         while(($char = fread($fd, 1)) != chr(0))
  118.             $ipAddr2 .= $char;
  119.  
  120.         $AddrSeek = implode('', unpack('L', $AddrSeek.chr(0)));
  121.         fseek($fd, $AddrSeek);
  122.  
  123.         while(($char = fread($fd, 1)) != chr(0))
  124.             $ipAddr1 .= $char;
  125.     } else {
  126.         fseek($fd, -1, SEEK_CUR);
  127.         while(($char = fread($fd, 1)) != chr(0))
  128.             $ipAddr1 .= $char;
  129.  
  130.         $ipFlag = fread($fd, 1);
  131.         if($ipFlag == chr(2)) {
  132.             $AddrSeek2 = fread($fd, 3);
  133.             if(strlen($AddrSeek2) < 3) {
  134.                 fclose($fd);
  135.                 return 'System Error';
  136.             }
  137.             $AddrSeek2 = implode('', unpack('L', $AddrSeek2.chr(0)));
  138.             fseek($fd, $AddrSeek2);
  139.         } else {
  140.             fseek($fd, -1, SEEK_CUR);
  141.         }
  142.         while(($char = fread($fd, 1)) != chr(0)){
  143.             $ipAddr2 .= $char;
  144.         }
  145.     }
  146.     fclose($fd);
  147.  
  148.     //最后做相应的替换操作后返回结果
  149.     if(preg_match('/http/i', $ipAddr2)) {
  150.         $ipAddr2 = '';
  151.     }
  152.     $ipaddr = "$ipAddr1 $ipAddr2";
  153.     $ipaddr = preg_replace('/CZ88.NET/is', '', $ipaddr);
  154.     $ipaddr = preg_replace('/^s*/is', '', $ipaddr);
  155.     $ipaddr = preg_replace('/s*$/is', '', $ipaddr);
  156.     if(preg_match('/http/i', $ipaddr) || $ipaddr == '') {
  157.         $ipaddr = 'Unknown';
  158.     }
  159.  
  160.     return $ipaddr;
  161. }
  162.  
  163.  
  164. //========================
  165. //
  166. //  调用举例(速度很快)
  167. //
  168. //========================
  169.  
  170. echo convertip('219.238.235.10');
  171. //输出: 北京市 电信通
  172.  
  173. echo convertip('23.56.82.12');
  174. //输出:IANA
  175.  
  176. echo convertip('250.69.52.0');
  177. //输出:IANA保留地址
  178.  
  179. echo convertip('238.69.52.0');
  180. //输出:IANA保留地址 用于多点传送
  181.  
  182. echo convertip('192.168.0.1');
  183. //输出:局域网 对方和您在同一内部网
  184.  
  185. echo convertip('255.255.255.255');
  186. //输出:纯真网络 2006年11月20日IP数据
  187.  
  188. ?>

附:(相应其他实现程序)

利用 QQWry.Dat 实现 IP 地址高效检索(PHP)(作者: andot)

纯真IP数据库(QQWry.Dat)查询 C源码 (作者:Windix)


感谢您的关注。您现在可以 留言(0)留下通告地址



Leave a Reply

Note: Any comments are permitted only because the site owner is letting you post, and any comments will be removed for any reason at the absolute discretion of the site owner.

*
To prove you're a person (not a spam script), type the security word shown in the picture.
Anti-Spam Image