欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP获取用户客户端真实IP的解决方案

程序员文章站 2024-04-02 13:17:28
获取客户端ip其实不是个简单的活儿,因为存在ip欺骗,和代理问题,所以获取客户端的ip的真实性会打折扣的,不能百分百准确.但是我们还是尽量找一个比较完善的获取客户端真正ip...

获取客户端ip其实不是个简单的活儿,因为存在ip欺骗,和代理问题,所以获取客户端的ip的真实性会打折扣的,不能百分百准确.但是我们还是尽量找一个比较完善的获取客户端真正ip方法.使用php获取ip的方法能找到很多.

function getip(){
if (getenv("http_client_ip") && strcasecmp(getenv("http_client_ip"), "unknown"))
$ip = getenv("http_client_ip");
else if (getenv("http_x_forwarded_for") && strcasecmp(getenv("http_x_forwarded_for"), "unknown"))
$ip = getenv("http_x_forwarded_for");
else if (getenv("remote_addr") && strcasecmp(getenv("remote_addr"), "unknown"))
$ip = getenv("remote_addr");
else if (isset($_server['remote_addr']) && $_server['remote_addr'] && strcasecmp($_server['remote_addr'], "unknown"))
$ip = $_server['remote_addr'];
else
$ip = "unknown";
return($ip);

现在需要对这段代码进行解释,这里用到了两个函数,getenv()和strcasecmp(),前一个函数获取得系统的环境变量,如果能取到值,则返回该值,不能则返回false.

$_server是服务器超级全局变量数组,用$_server['remote_addr']同样可以获取到客户端的ip地址.二者的区别在于,getenv不支持iis的isapi方式运行的php.

strcasecmp(string1,string2)字符串函数的用法是把string1和string2进行比较,如果相等返回0,如果string1大于string2,返回大于0的数,小于则返回小于0的数.

函数先使用客户ip,如果不成立尝试用代理的方法,如果不行,再使用remote_addr.

还看到过一个检测ip更详细的方法,考虑了ip的欺骗,和多重代理代码.方法相类似.

function getip() {
$unknown = 'unknown';
if ( isset($_server['http_x_forwarded_for']) && $_server['http_x_forwarded_for'] && strcasecmp($_server['http_x_forwarded_for'], $unknown) ) {
$ip = $_server['http_x_forwarded_for'];
} elseif ( isset($_server['remote_addr']) && $_server['remote_addr'] && strcasecmp($_server['remote_addr'], $unknown) ) {
$ip = $_server['remote_addr'];
}
/*
处理多层代理的情况
或者使用正则方式:$ip = preg_match("/[\d\.]{7,15}/", $ip, $matches) ? $matches[0] : $unknown;
*/
if (false !== strpos($ip, ','))
$ip = reset(explode(',', $ip));
return $ip;
}

一、没有使用代理服务器的php获取客户端ip情况:

remote_addr = 客户端ip
http_x_forwarded_for = 没数值或不显示

二、使用透明代理服务器的情况:transparent proxies

remote_addr = 最后一个代理服务器 ip
http_x_forwarded_for = 客户端真实 ip (经过多个代理服务器时,这个值类似:221.5.252.160, 203.98.182.163, 203.129.72.215)

这类代理服务器还是将客户端真实的ip发送给了访问对象,无法达到隐藏真实身份的目的.

三、使用普通匿名代理服务器的php获取客户端ip情况:anonymous proxies

remote_addr = 最后一个代理服务器 ip
http_x_forwarded_for = 代理服务器 ip (经过多个代理服务器时,这个值类似:203.98.182.163, 203.98.182.163, 203.129.72.215)

这种情况下隐藏了客户端的真实ip,但是向访问对象透露了客户端是使用代理服务器访问它们的.

四、使用欺骗性代理服务器的情况:distorting proxies

remote_addr = 代理服务器 ip
http_x_forwarded_for = 随机的 ip(经过多个代理服务器时,这个值类似:220.4.251.159, 203.98.182.163, 203.129.72.215)

这种情况下同样透露了客户端是使用了代理服务器,但编造了一个虚假的随机ip(220.4.251.159)代替客户端的真实ip来欺骗它.

五、使用高匿名代理服务器的php获取客户端ip情况:high anonymity proxies (elite proxies)

remote_addr = 代理服务器 ip

http_x_forwarded_for = 没数值或不显示

无论是remote_addr还是http_forwarded_for,这些头消息未必能够取得到,因为不同的浏览器不同的网络设备可能发送不同的ip头消息.因此php使用$_server["remote_addr"] 、$_server["http_x_forwarded_for"] 获取的值可能是空值也可能是“unknown”值.

以上所述是小编给大家介绍的php获取用户客户端真实ip的解决方案,希望对大家有所帮助