基于PHP获取服务器访客IP地址

本文介绍基于PHP实现获取Web服务器访客的IP地址

index.php代码

<!doctype html>
<html>
<head>
    <meta charset="utf-8">
    <title>Index</title>
    <style>
        .container {
            width: 60%;
            margin: 10% auto 0;
            background-color: #f0f0f0;
            padding: 2% 5%;
            border-radius: 10px
        }

        ul {
            padding-left: 20px;
        }

            ul li {
                line-height: 2.3
            }

        a {
            color: #20a53a
        }
    </style>
</head>
<body>
    <div class="container">
        <h1>Welcome</h1><hr/>
    <h2>
        <?php
            $str1="当前来访IP地址:";
            function get_real_ip(){
                if(getenv("HTTP_CLIENT_IP") && strcasecmp(getenv("HTTP_CLIENT_IP"), "unknow")){
                    $ip = getenv("HTTP_CLIENT_IP");
                }else if(getenv("HTTP_X_FORWARDED_FOR") && strcasecmp(getenv("HTTP_X_FORWARDED_FOR"), "unknow")){
                    $ip = getenv("HTTP_X_FORWARDED_FOR");
                }else if(getenv("REMOTE_ADDR") && strcasecmp(getenv("REMOTE_ADDR"), "unknow")){
                    $ip = getenv("REMOTE_ADDR");
                }else if(isset($_SERVER["REMOTE_ADDR"]) && $_SERVER["REMOTE_ADDR"] && strcasecmp($_SERVER["REMOTE_ADDR"],"unknow")){
                    $ip = $_SERVER["REMOTE_ADDR"];
                }else{
                    $ip = "unknow";
                }
                return $ip;
            }
            echo $str1 . " " . get_real_ip();
        ?>
        <?php
            //CMCC_range1_start
            $cmcc_lanip1_start = get_iplong('172.16.0.2'); 
            //CMCC_range1_end
            $cmcc_lanip1_end = get_iplong('172.21.255.255');
            //CMCC_range2_start
            $cmcc_lanip2_start = get_iplong('172.24.0.1'); 
            //CMCC_range2_end
            $cmcc_lanip2_end = get_iplong('172.31.255.255');
            //Unicom_start
            $unicom_lanip_start = get_iplong('172.22.0.0'); 
            //Unicom_end
            $unicom_lanip_end = get_iplong('172.22.255.255');
            //telecom_start
            $telecom_lanip_start = get_iplong('172.23.0.0'); 
            //telecom_end
            $telecom_lanip_end = get_iplong('172.23.255.255');
            //特定IP
            //移动网关IP
            $cmcc_gateway = get_iplong('172.21.0.1');
            //联通网关IP
            $unicom_gateway = get_iplong('172.22.0.1');
            //电信网关IP
            $telecom_gateway = get_iplong('172.23.0.1');
            //待判断的ip
            $ipaddr = get_iplong(get_real_ip());
            //开始判断
            switch ($ipaddr) {
                case $ipaddr == $cmcc_gateway:
                    echo '<br/><a style="color:#FF4500;">(移动内网访问)</a>';
                    break;
                case $ipaddr == $unicom_gateway:
                    echo '<br/><a style="color:#FF4500;">(联通内网访问)</a>';
                    break;
                case $ipaddr == $telecom_gateway:
                    echo '<br/><a style="color:#FF4500;">(电信内网访问)</a>';
                    break;
                case $ipaddr >= $cmcc_lanip1_start && $ipaddr <= $cmcc_lanip1_end:
                    echo '(移动出口)';
                    break;
                case $ipaddr >= $cmcc_lanip2_start && $ipaddr <= $cmcc_lanip2_end:
                    echo '(移动出口)';
                    break;
                case $ipaddr >= $unicom_lanip_start && $ipaddr <= $unicom_lanip_end:
                    echo '(联通出口)';
                    break;
                case $ipaddr >= $telecom_lanip_start && $ipaddr <= $telecom_lanip_end:
                    echo '(电信出口)';
                    break;
                default:
                    echo '(公网访问)';
            }
            include("qqwry.php"); // 引入代码
            $index = new QQWry('qqwry.dat'); // 初始化类
            $detail = $index->getDetail(get_real_ip()); // 调用查询函数
            ?>
            <br/><hr/>
            <?php
            $part_detail = array_slice($detail,2,3);
            $ip_loc = implode(" ",$part_detail);?>
            <h2>IP归属信息:<?php echo $ip_loc;?></h2>
            <hr/>
            <?php
            echo '<h2>IP数据库更新日期:' . $index->getVersion() . PHP_EOL . '</h2>';
            //转换IP地址
            function get_iplong($ipaddr){
                return bindec(decbin(ip2long($ipaddr)));
            }
        ?>
    </h2>
    </div>
</body>
</html>

qqwry.php代码

<?php
// 数据来源:纯真IP数据库 qqwry.dat
// 初始化类:new QQWry($fileName)
// 请求方式:getDetail($ip)
// 返回格式:
// {
//     "beginIP": IP段起始点
//     "endIP": IP段结束点
//     "dataA": 数据段1
//     "dataB": 数据段2
// }
// 
// 请求版本:getVersion()
// 返回格式:YYYYMMDD

class QQWry {
    private $fp; // 文件指针
    private $firstRecord; // 第一条记录的偏移地址
    private $lastRecord; // 最后一条记录的偏移地址
    private $recordNum; // 总记录条数

    public function __construct($fileName) { // 构造函数
        $this->fp = fopen($fileName, 'rb');
        $this->firstRecord = $this->read4byte();
        $this->lastRecord  = $this->read4byte();
        $this->recordNum = ($this->lastRecord - $this->firstRecord) / 7; // 每条索引长度为7字节
    }

    public function __destruct() { // 析构函数
        if ($this->fp) {
            fclose($this->fp);
        }
    }

    private function read4byte() { // 读取4字节并转为long
        return unpack('Vlong', fread($this->fp, 4))['long'];
    }

    private function read3byte() { // 读取3字节并转为long
        return unpack('Vlong', fread($this->fp, 3) . chr(0))['long'];
    }

    private function readString() { // 读取字符串
        $str = '';
        $char = fread($this->fp, 1);
        while (ord($char) != 0) { // 读到二进制0结束
            $str .= $char;
            $char = fread($this->fp, 1);
        }
        return $str;
    }

    private function zipIP($ip) { // IP地址转为数字
        $ip_arr = explode('.', $ip);
        $tmp = (16777216 * intval($ip_arr[0])) + (65536 * intval($ip_arr[1])) + (256 * intval($ip_arr[2])) + intval($ip_arr[3]);
        return pack('N', intval($tmp)); // 32位无符号大端序长整型
    }

    private function unzipIP($ip) { // 数字转为IP地址
        return long2ip($ip);
    }

    public function getVersion() { // 获取当前数据库的版本
        fseek($this->fp, $this->lastRecord + 4);
        $tmp = $this->getRecord($this->read3byte())['B'];
        return substr($tmp, 0, 4) . substr($tmp, 7, 2) . substr($tmp, 12, 2);
    }

    public function getDetail($ip) { // 获取IP地址区段及所在位置
        if (!filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4)) { // 判断是否为IPv4地址
            return null;
        }

        fseek($this->fp, $this->searchRecord($ip)); // 跳转到对应IP记录的位置
        $detail['beginIP'] = $this->unzipIP($this->read4byte()); // 目标IP所在网段的起始IP
        $offset = $this->read3byte(); // 索引后3字节为对应记录的偏移量
        fseek($this->fp, $offset);
        $detail['endIP'] = $this->unzipIP($this->read4byte()); // 目标IP所在网段的结束IP

        $tmp = $this->getRecord($offset); // 获取记录的dataA与dataB
        $detail['dataA'] = $tmp['A'];
        $detail['dataB'] = $tmp['B'];

        if ($detail['beginIP'] == '255.255.255.0') { // 去除附加信息
            $detail['dataA'] = 'IANA';
            $detail['dataB'] = '保留地址';
        }
        if ($detail['dataA'] == ' CZ88.NET' || $detail['dataA'] == '纯真网络') {
            $detail['dataA'] = '';
        }
        if ($detail['dataB'] == ' CZ88.NET') {
            $detail['dataB'] = '';
        }
        return $detail;
    }

    private function searchRecord($ip) { // 根据IP地址获取索引的绝对偏移量
        $ip = $this->zipIP($ip); // 转为数字以比较大小
        $down = 0;
        $up = $this->recordNum;
        while ($down <= $up) { // 二分法查找
            $mid = floor(($down + $up) / 2); // 计算二分点
            fseek($this->fp, $this->firstRecord + $mid * 7);
            $beginip = strrev(fread($this->fp, 4)); // 获取二分区域的下边界
            if ($ip < $beginip) { // 目标IP在二分区域以下
                $up = $mid - 1; // 缩小搜索的上边界
            } else {
                fseek($this->fp, $this->read3byte());
                $endip = strrev(fread($this->fp, 4)); // 获取二分区域的上边界
                if ($ip > $endip) { // 目标IP在二分区域以上
                    $down = $mid + 1; // 缩小搜索的下边界
                } else { // 目标IP在二分区域内
                    return $this->firstRecord + $mid * 7; // 返回索引的偏移量
                }
            }
        }
        return $this->lastRecord; // 无法找到对应索引,返回最后一条记录的偏移量
    }

    private function getRecord($offset) { // 读取IP记录的数据
        fseek($this->fp, $offset + 4);
        $flag = ord(fread($this->fp, 1));
        if ($flag == 1) { // dataA与dataB均重定向
            $offset = $this->read3byte(); // 重定向偏移
            fseek($this->fp, $offset);
            if (ord(fread($this->fp, 1)) == 2) { // dataA再次重定向
                fseek($this->fp, $this->read3byte());
                $data['A'] = $this->readString();
                fseek($this->fp, $offset + 4);
                $data['B'] = $this->getDataB();
            } else { // dataA无重定向
                fseek($this->fp, -1, SEEK_CUR); // 文件指针回退1字节
                $data['A'] = $this->readString();
                $data['B'] = $this->getDataB();
            }
        } else if ($flag == 2) { // dataA重定向
            fseek($this->fp, $this->read3byte());
            $data['A'] = $this->readString();
            fseek($this->fp, $offset + 8); // IP占4字节, 重定向标志占1字节, dataA指针占3字节
            $data['B'] = $this->getDataB();
        } else { // dataA无重定向
            fseek($this->fp, -1, SEEK_CUR); // 文件指针回退1字节
            $data['A'] = $this->readString();
            $data['B'] = $this->getDataB();
        }
        $data['A'] = iconv("GBK", "UTF-8", $data['A']); // GBK -> UTF-8
        $data['B'] = iconv("GBK", "UTF-8", $data['B']);
        return $data;
    }

    private function getDataB() { // 从fp指定偏移获取dataB
        $flag = ord(fread($this->fp, 1));
        if ($flag == 0) { // dataB无信息
            return '';
        } else if ($flag == 1 || $flag == 2) { // dataB重定向
            fseek($this->fp, $this->read3byte());
            return $this->readString();
        } else { // dataB无重定向
            fseek($this->fp, -1, SEEK_CUR); // 文件指针回退1字节
            return $this->readString();
        }
    }
}
?>

演示结果

IP数据库文件qqwry.dat

(数据库更新日期:20220223)

注:文件为.zip格式,下载完成后需解压使用

来源:默认网盘

版权声明:
作者:admin
链接:https://luhaoyu.com/index.php/2022/03/11/%e5%9f%ba%e4%ba%8ephp%e8%8e%b7%e5%8f%96%e6%9c%8d%e5%8a%a1%e5%99%a8%e8%ae%bf%e5%ae%a2ip%e5%9c%b0%e5%9d%80/
来源:悄悄拔尖
文章版权归作者所有,未经允许请勿转载。

THE END
分享
二维码
< <上一篇
下一篇>>
文章目录
关闭
目 录