1.请问使用php可以生成ip数据库的.dat文件吗?求方法2.求php能够遍历出.dat的ip数据库的程序,比如QQwry.dat。

解决方案 »

  1.   

    这是个PHP版读取纯真IP数据库(QQWry.dat)实例。
    以下函数conrvertip()位于 Discuz!5_GBK/upload/include/misc.func.Php 路径中,有兴趣可以具体去阅读分析。(下面代码我做了简单的修改,更便于阅读,核心没有修改)
    view sourceprint?
    001
    <span style="font-size:16px;"><span style="font-family:comic sans ms,cursive;"><?
    002
    //===================================
    003
    //
    004
    // 功能:IP地址获取真实地址函数
    005
    // 参数:$ip - IP地址
    006
    // 作者:[Discuz!] (C) Comsenz Inc.
    007
    //
    008
    //===================================
    009
    function convertip($ip) {
    010
        //IP数据文件路径
    011
        $dat_path = 'QQWry.Dat';
    012
     
    013
        //检查IP地址
    014
        if(!preg_match("/^d{1,3}.d{1,3}.d{1,3}.d{1,3}$/", $ip)) {
    015
            return 'IP Address Error';
    016
        }
    017
        //打开IP数据文件
    018
        if(!$fd = @fopen($dat_path, 'rb')){
    019
            return 'IP date file not exists or access denied';
    020
        }
    021
     
    022
        //分解IP进行运算,得出整形数
    023
        $ip = explode('.', $ip);
    024
        $ipNum = $ip[0] * 16777216 + $ip[1] * 65536 + $ip[2] * 256 + $ip[3];
    025
     
    026
        //获取IP数据索引开始和结束位置
    027
        $DataBegin = fread($fd, 4);
    028
        $DataEnd = fread($fd, 4);
    029
        $ipbegin = implode('', unpack('L', $DataBegin));
    030
        if($ipbegin < 0) $ipbegin += pow(2, 32);
    031
        $ipend = implode('', unpack('L', $DataEnd));
    032
        if($ipend < 0) $ipend += pow(2, 32);
    033
        $ipAllNum = ($ipend - $ipbegin) / 7 + 1;
    034
        
    035
        $BeginNum = 0;
    036
        $EndNum = $ipAllNum;
    037
     
    038
        //使用二分查找法从索引记录中搜索匹配的IP记录
    039
        while($ip1num>$ipNum || $ip2num<$ipNum) {
    040
            $Middle= intval(($EndNum + $BeginNum) / 2);
    041
     
    042
            //偏移指针到索引位置读取4个字节
    043
            fseek($fd, $ipbegin + 7 * $Middle);
    044
            $ipData1 = fread($fd, 4);
    045
            if(strlen($ipData1) < 4) {
    046
                fclose($fd);
    047
                return 'System Error';
    048
            }
    049
            //提取出来的数据转换成长整形,如果数据是负数则加上2的32次幂
    050
            $ip1num = implode('', unpack('L', $ipData1));
    051
            if($ip1num < 0) $ip1num += pow(2, 32);
    052
            
    053
            //提取的长整型数大于我们IP地址则修改结束位置进行下一次循环
    054
            if($ip1num > $ipNum) {
    055
                $EndNum = $Middle;
    056
                continue;
    057
            }
    058
            
    059
            //取完上一个索引后取下一个索引
    060
            $DataSeek = fread($fd, 3);
    061
            if(strlen($DataSeek) < 3) {
    062
                fclose($fd);
    063
                return 'System Error';
    064
            }
    065
            $DataSeek = implode('', unpack('L', $DataSeek.chr(0)));
    066
            fseek($fd, $DataSeek);
    067
            $ipData2 = fread($fd, 4);
    068
            if(strlen($ipData2) < 4) {
    069
                fclose($fd);
    070
                return 'System Error';
    071
            }
    072
            $ip2num = implode('', unpack('L', $ipData2));
    073
            if($ip2num < 0) $ip2num += pow(2, 32);
    074
     
    075
            //没找到提示未知
    076
            if($ip2num < $ipNum) {
    077
                if($Middle == $BeginNum) {
    078
                    fclose($fd);
    079
                    return 'Unknown';
    080
                }
    081
                $BeginNum = $Middle;
    082
            }
    083
        }
    084
     
    085
        //下面的代码读晕了,没读明白,有兴趣的慢慢读
    086
        $ipFlag = fread($fd, 1);
    087
        if($ipFlag == chr(1)) {
    088
            $ipSeek = fread($fd, 3);
    089
            if(strlen($ipSeek) < 3) {
    090
                fclose($fd);
    091
                return 'System Error';
    092
            }
    093
            $ipSeek = implode('', unpack('L', $ipSeek.chr(0)));
    094
            fseek($fd, $ipSeek);
    095
            $ipFlag = fread($fd, 1);
    096
        }
    097
     
    098
        if($ipFlag == chr(2)) {
    099
            $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
    echo convertip('219.238.235.10');
    170
    //输出: 北京市 电信通
    171
     
    172
    echo convertip('23.56.82.12');
    173
    //输出:IANA
    174
     
    175
    echo convertip('250.69.52.0');
    176
    //输出:IANA保留地址
    177
     
    178
    echo convertip('238.69.52.0');
    179
    //输出:IANA保留地址 用于多点传送
    180
     
    181
    echo convertip('192.168.0.1');
    182
    //输出:局域网 对方和您在同一内部网
    183
     
    184
    echo convertip('255.255.255.255');
    185
    //输出:纯真网络 2006年11月20日IP数据
    186
     
    187
    ?>
    188
    </span></span>
      

  2.   

    <span style="font-size:16px;"><span style="font-family:comic sans ms,cursive;"><?
    //===================================
    //
    // 功能: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');
    //输出: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数据?>
    </span></span>
      

  3.   

    我主要想问的一个问题是,能够用php将普通的数据库或者文本文件生成类似的ip数据库dat文件么具体的实现思路或者实现方法能讲解一下吗
      

  4.   

    我的意思是想知道 php通过什么方法 能够将IP地址以及其信息编码成为16进制码显示,不知道这样表达是否正确。