设为首页收藏本站

就爱编程论坛

 找回密码
 注册

人人连接登陆

无需注册,直接登录

用新浪微博连接

一步搞定

QQ登录

只需一步,快速开始

查看: 435|回复: 0
打印 上一主题 下一主题

PHP读取纯真IP数据库的函数 [复制链接]

Rank: 9Rank: 9Rank: 9

  • TA的每日心情
    无聊
    2025-5-27 03:37:20
  • 签到天数: 366 天

    [LV.9]以坛为家II

    论坛先锋 学习至圣 荣誉成员 论坛元老 活跃之星 终极领袖

    我玩的应用:

    跳转到指定楼层
    楼主
    发表于 2011-4-24 06:48:41 |只看该作者 |倒序浏览
    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 路径中,有兴趣可以具体去阅读分析。
    <?php//===================================
    //
    // 功能:IP地址获取真实地址函数
    // 参数:$ip - IP地址
    // 作者:[Discuz!] (C) Comsenz Inc.
    //
    //===================================
    //
    // 调用举例(速度很快)
    //
    //========================echo convertip('219.238.235.10');
    //输出: 北京市 电信通echo convertip('23.56.82.12');
    //输出:IANAecho 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数据
    $ip='210.32.0.0';
    echo convertip($ip);
    #Notice: Undefined variable: ip1num in E:\phpLearn\IPcount\index.php on line 81
    #Notice: Undefined variable: ip2num in E:\phpLearn\IPcount\index.php on line 81
    #Notice: Undefined variable: ipAddr2 in E:\phpLearn\IPcount\index.php on line 150
    #Notice: Undefined variable: ipAddr1 in E:\phpLearn\IPcount\index.php on line 152
    #福建省福州市 铁通
    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 地址错误!';
        }
        //打开IP数据文件
        if(!$fd = @fopen($dat_path, 'rb')){
            return 'IP数据文件无法读取,请确保是正确的纯真IP库!';
        }    //分解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)); //unpack() 函数从二进制字符串对数据进行解包。unpack(format,data) L - unsigned long (always 32 bit, machine byte order)
    #$ipbegin 值如:5386001
    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记录
    $ip1num=''; $ip2num='';   $ipAddr1='';    $ipAddr2='';
        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;
    }
    ?>
    附:(相应其他实现程序)
    注:www.coolcode.cn如不能访问,改成:www.coolcode.orgPhp)" href="http://www.coolcode.cn/?action=search&searchid=473" rel=bookmark>利用 QQWry.Dat 实现 IP 地址高效检索(Php)(作者: andot)
    或查看:
    利用 QQWry.Dat 实现 IP 地址高效检索(PHP)
    http://hi.baidu.com/traindiy/blog/item/a37edb585fe8bb8a810a183b.html
    利用 QQWry.Dat 实现 IP 地址高效检索(PHP 共享内存版)
    http://hi.baidu.com/traindiy/blog/item/be3733db73a9856dd1164e38.html
    分享到: QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    分享分享0 收藏收藏0 支持支持0 反对反对0 分享到人人 转发到微博
    [img=http://mail.qq.com/cgi-bin/qm_share?t=qm_mailme&email=fRUcHhYWGAQ9GxIFEBwUEVMeEhA]http://rescdn.qqmail.com/zh_CN/htmledition/images/function/qm_open/ico_mailme_02.png[/img]

    使用道具 举报

    您需要登录后才可以回帖 登录 | 注册 人人连接登陆

    晴云孤魂's Blog|就爱编程搜帖|手机版|Archiver|就爱编程论坛     

    GMT+8, 2025-7-2 00:42 , Processed in 0.099065 second(s), 27 queries .

    Powered by Discuz! X2

    © 2001-2011 Comsenz Inc.

    回顶部