PHP获取真实IP地址的正确方法
积累知识,胜过积蓄金银!毕竟在文章开发的过程中,会遇到各种各样的问题,往往都是一些细节知识点还没有掌握好而导致的,因此基础知识点的积累是很重要的。下面本文《PHP获取用户真实IP地址方法》,就带大家讲解一下知识点,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~
答案:在PHP中,$_SERVER['REMOTE_ADDR']不总是真实IP,因为当请求经过CDN、负载均衡或反向代理时,该值仅为代理服务器IP。真实客户端IP通常由代理服务器通过HTTP头如X-Forwarded-For或X-Real-IP传递。为获取可信IP,应优先解析这些头信息,并结合信任代理IP范围和IP有效性验证(如过滤私有IP),防止IP欺骗。最安全的做法是在Web服务器(如Nginx)层面配置real_ip_header和set_real_ip_from,将真实IP写入REMOTE_ADDR,PHP直接读取即可。

PHP中获取客户端IP地址,最直接的方法是使用$_SERVER['REMOTE_ADDR']。然而,在现代复杂的网络环境中,比如网站部署在CDN、负载均衡器或反向代理之后,REMOTE_ADDR往往只能获取到代理服务器的IP,而非用户的真实IP。这时,我们需要检查一系列HTTP头信息,如X-Forwarded-For、X-Real-IP等,来尝试还原用户的真实IP地址。
解决方案
要获取客户端的真实IP,我们需要一个综合性的策略,不能仅仅依赖$_SERVER['REMOTE_ADDR']。原因很简单,当用户请求经过代理服务器(比如CDN、负载均衡器、Nginx反向代理等)时,REMOTE_ADDR记录的是与你的服务器直接通信的那个设备的IP,也就是代理服务器的IP。用户的真实IP,通常会被代理服务器放在特定的HTTP请求头中转发过来。
以下是一个相对健壮的PHP函数,用于尝试获取用户的真实IP地址:
function getClientIp() {
$ip = '';
// 优先检查 X-Forwarded-For,因为它通常包含更远的客户端IP
if (isset($_SERVER['HTTP_X_FORWARDED_FOR']) && !empty($_SERVER['HTTP_X_FORWARDED_FOR'])) {
$ip = trim(explode(',', $_SERVER['HTTP_X_FORWARDED_FOR'])[0]); // X-Forwarded-For 可能是逗号分隔的列表,取第一个
}
// 如果 X-Forwarded-For 不存在或为空,检查 X-Real-IP
if (empty($ip) && isset($_SERVER['HTTP_X_REAL_IP']) && !empty($_SERVER['HTTP_X_REAL_IP'])) {
$ip = trim($_SERVER['HTTP_X_REAL_IP']);
}
// 如果以上都获取不到,退回到 REMOTE_ADDR
if (empty($ip) && isset($_SERVER['REMOTE_ADDR']) && !empty($_SERVER['REMOTE_ADDR'])) {
$ip = trim($_SERVER['REMOTE_ADDR']);
}
// 最后,进行IP地址的有效性验证和私有IP过滤
// 这是一个关键步骤,防止IP欺骗和获取到内部IP
if (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE) === false) {
// 如果不是有效的公共IP,尝试从 X-Forwarded-For 列表中寻找下一个
// 或者直接返回 REMOTE_ADDR (如果它不是私有IP)
// 这一步可以根据具体需求调整,这里简化为如果检测到非公共IP,就直接返回 REMOTE_ADDR
if (isset($_SERVER['REMOTE_ADDR']) && filter_var($_SERVER['REMOTE_ADDR'], FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE) !== false) {
return trim($_SERVER['REMOTE_ADDR']);
} else {
return '0.0.0.0'; // 或者其他默认值,表示无法获取有效IP
}
}
return $ip;
}
// 使用示例
$userIp = getClientIp();
// echo "您的IP地址是: " . $userIp;这个函数首先检查X-Forwarded-For,因为在多层代理下,它往往包含了最原始的客户端IP。然后是X-Real-IP,最后才是REMOTE_ADDR。重要的是,它还包含了IP地址的有效性验证和私有IP地址的过滤,这能有效避免一些IP欺骗的风险。
PHP获取客户端IP地址时,为什么$_SERVER['REMOTE_ADDR']不总是真实IP?
说实话,这个问题我刚开始接触Web开发的时候也困惑了很久。我们总觉得$_SERVER['REMOTE_ADDR']就是用户的IP,毕竟名字听起来就是“远程地址”嘛。但实际情况远比这复杂。
想象一下,你的网站部署在服务器A上。如果用户直接通过浏览器访问服务器A,那么REMOTE_ADDR确实就是用户的真实IP。这很好理解,因为用户浏览器是直接和服务器A建立TCP连接的。
然而,现在的网站架构很少这么简单粗暴了。大多数网站为了性能、安全、高可用性,都会在用户和服务器之间加入各种“中间人”:
- CDN (内容分发网络): 比如Cloudflare、阿里云CDN。用户访问你的域名,实际上是先连接到离他最近的CDN节点。CDN节点再从你的源站获取内容并缓存,或者直接转发请求。这时,你的服务器收到的请求,
REMOTE_ADDR就是CDN节点的IP,而不是用户的真实IP。 - 负载均衡器 (Load Balancer): 比如Nginx、HAProxy、或者云服务商提供的ELB/SLB。它们负责将流量分发到后端多台Web服务器上。用户请求先到达负载均衡器,负载均衡器再转发给后端服务器。同样,后端服务器看到的
REMOTE_ADDR是负载均衡器的IP。 - 反向代理 (Reverse Proxy): Nginx、Apache也可以作为反向代理。它们接收外部请求,然后转发给内部的Web服务(可能是PHP-FPM、Node.js应用等)。这种情况下,你的PHP应用看到的
REMOTE_ADDR是反向代理服务器的IP。
这些“中间人”在转发请求时,通常会很“好心”地把原始客户端的IP地址,通过自定义的HTTP头(最常见的就是X-Forwarded-For和X-Real-IP)添加到请求中,再发送给后端服务器。所以,如果你只看REMOTE_ADDR,你就只能看到这些中间人的IP,而不是真正的用户IP。这对于日志分析、用户行为追踪、地域限制等功能来说,都是一个巨大的障碍。
如何有效防范IP欺骗,确保获取到的是可信的客户端IP?
IP欺骗是一个很现实的安全问题,如果你的应用仅仅信任HTTP头里传过来的IP,那么攻击者很容易就能伪造IP地址,进行各种恶意操作,比如绕过基于IP的访问控制、伪造用户来源进行刷票或DDoS攻击等。要有效防范IP欺骗,并确保获取到的是可信的客户端IP,我们需要几个层面的思考和实践。
理解代理链与
X-Forwarded-For的结构:X-Forwarded-For头可能包含多个IP地址,用逗号分隔,格式通常是client_ip, proxy1_ip, proxy2_ip。按照约定,最左边的IP是最初的客户端IP,后续的是各个代理服务器的IP。但关键在于,攻击者可以轻易地在请求中伪造X-Forwarded-For头,将一个假IP放在最左边。信任已知代理: 这是最核心的防御策略。如果你的网站部署在CDN、负载均衡器或反向代理之后,并且你知道这些代理服务器的IP地址范围,那么你只应该信任这些代理服务器发出的
X-Forwarded-For或X-Real-IP头。配置Web服务器: 比如Nginx,可以通过
set_real_ip_from指令来指定信任的代理IP范围,然后real_ip_header指令来指定从哪个头获取真实IP。这样,Nginx会将真实的客户端IP(从受信任的代理头中解析出来的)覆盖到REMOTE_ADDR变量中,PHP应用就能直接通过$_SERVER['REMOTE_ADDR']获取到“真实”IP了,因为它已经被Nginx处理过了。# 信任Cloudflare的IP范围,或者你自己的负载均衡器IP set_real_ip_from 103.21.244.0/22; set_real_ip_from 103.22.200.0/22; # ... 更多信任IP范围 real_ip_header X-Forwarded-For; # 从这个头获取真实IP real_ip_recursive on; # 递归处理X-Forwarded-For,直到找到第一个非信任IP
这样配置后,
$_SERVER['REMOTE_ADDR']将是经过Nginx处理后的真实客户端IP,或者最近一个不受信任的代理IP。在PHP代码中过滤: 如果你无法控制Web服务器的配置,或者需要更细粒度的控制,可以在PHP代码中实现。
function getTrustedClientIp() { $ip = '0.0.0.0'; $trustedProxies = [ '192.168.1.1', // 你的负载均衡器IP '10.0.0.0/8', // 你的内部网络范围 // ... 更多你信任的代理IP或IP段 ]; $remoteAddr = $_SERVER['REMOTE_ADDR'] ?? ''; // 检查 REMOTE_ADDR 是否在信任列表中 $isTrustedProxy = false; foreach ($trustedProxies as $proxy) { if (strpos($proxy, '/') !== false) { // CIDR if (filter_var($remoteAddr, FILTER_VALIDATE_IP) && cidr_match($remoteAddr, $proxy)) { $isTrustedProxy = true; break; } } else { // 单个IP if ($remoteAddr === $proxy) { $isTrustedProxy = true; break; } } } // 如果 REMOTE_ADDR 是一个受信任的代理,那么我们相信 X-Forwarded-For if ($isTrustedProxy && isset($_SERVER['HTTP_X_FORWARDED_FOR']) && !empty($_SERVER['HTTP_X_FORWARDED_FOR'])) { $forwardedIps = array_map('trim', explode(',', $_SERVER['HTTP_X_FORWARDED_FOR'])); // 遍历 X-Forwarded-For 列表,找到第一个非私有、非保留的IP foreach ($forwardedIps as $fwdIp) { if (filter_var($fwdIp, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE)) { $ip = $fwdIp; break; } } } else { // 如果 REMOTE_ADDR 不是受信任的代理,或者没有 X-Forwarded-For, // 那么 REMOTE_ADDR 就是我们能得到的“最真实”的IP了 // 但仍需过滤私有/保留IP if (filter_var($remoteAddr, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE)) { $ip = $remoteAddr; } } return $ip; } // 辅助函数:检查IP是否在CIDR范围内 function cidr_match($ip, $cidr) { list($subnet, $mask) = explode('/', $cidr); if ((ip2long($ip) & ~((1 << (32 - $mask)) - 1)) == ip2long($subnet)) { return true; } return false; }这个函数复杂一些,但更安全。它只有在确认
REMOTE_ADDR是已知代理的IP时,才会去解析X-Forwarded-For。否则,它就直接使用REMOTE_ADDR(前提是它不是私有IP)。
过滤私有IP和保留IP: 无论从哪个头获取IP,都要对其进行
filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE)这样的验证。这可以确保IP不是10.x.x.x、192.168.x.x、172.16.x.x这类内部网络IP,也不是127.0.0.1这类本地回环地址。如果获取到的IP是这些,那它肯定不是你想要的客户端真实IP。日志记录与监控: 即使有了防御措施,也应该记录所有相关的IP信息,包括
REMOTE_ADDR、X-Forwarded-For、X-Real-IP等。这对于后续的审计和安全分析非常有帮助。如果发现异常模式(比如某个IP突然从多个不同地理位置的代理IP发送请求),可以及时发现并处理。
总而言之,获取客户端IP不是一个简单的“取值”问题,而是一个涉及网络架构、安全考量和代码健壮性的综合工程。信任链的建立,以及对IP地址的严格验证,是确保获取到可信IP的关键。
好了,本文到此结束,带大家了解了《PHP获取真实IP地址的正确方法》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!
MBTI人格类型全解析与性格特征
- 上一篇
- MBTI人格类型全解析与性格特征
- 下一篇
- Kimi积分怎么查?余额查看方法详解
-
- 文章 · php教程 | 1小时前 |
- PHP实现前后端分离实战技巧
- 274浏览 收藏
-
- 文章 · php教程 | 1小时前 |
- SVG转PNG的PHPImagick替代方案
- 356浏览 收藏
-
- 文章 · php教程 | 2小时前 |
- Woocommerce跳转结账页设置教程
- 490浏览 收藏
-
- 文章 · php教程 | 2小时前 |
- PHP数组追加元素:array\_push与\[\]对比
- 369浏览 收藏
-
- 文章 · php教程 | 3小时前 |
- PhpStorm启用Emmet设置教程详解
- 490浏览 收藏
-
- 文章 · php教程 | 3小时前 |
- PHP链接嵌入失败解决方法详解
- 450浏览 收藏
-
- 文章 · php教程 | 3小时前 | php
- PHP资源句柄检测与释放技巧
- 492浏览 收藏
-
- 文章 · php教程 | 3小时前 |
- PHP无索引数组处理方法及array_values使用教程
- 158浏览 收藏
-
- 文章 · php教程 | 3小时前 |
- 双击PHP文件无法打开解决方法
- 143浏览 收藏
-
- 文章 · php教程 | 4小时前 |
- PHP文件无法查看代码的常见原因及解决方法
- 141浏览 收藏
-
- 文章 · php教程 | 4小时前 |
- PHP获取真实IP的正确方法及常见问题
- 243浏览 收藏
-
- 文章 · php教程 | 5小时前 |
- PHP创建微信菜单数组详细教程
- 204浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3527次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3756次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3752次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4896次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 4118次使用
-
- PHP技术的高薪回报与发展前景
- 2023-10-08 501浏览
-
- 基于 PHP 的商场优惠券系统开发中的常见问题解决方案
- 2023-10-05 501浏览
-
- 如何使用PHP开发简单的在线支付功能
- 2023-09-27 501浏览
-
- PHP消息队列开发指南:实现分布式缓存刷新器
- 2023-09-30 501浏览
-
- 如何在PHP微服务中实现分布式任务分配和调度
- 2023-10-04 501浏览

