$_SERVER['SERVER_ADDR'] 返回的“乱码”通常是IPv6地址(如::1)被误当UTF-8显示所致,并非真乱码;真正编码问题多源于exec('ipconfig')等外部命令输出与PHP默认UTF-8不匹配。

PHP $_SERVER['SERVER_ADDR'] 返回乱码?先确认是不是真乱码
绝大多数情况,$_SERVER['SERVER_ADDR'] 或 gethostbyname(gethostname()) 返回的不是“乱码”,而是 IPv6 地址(如 ::1 或 fe80::1)被错误当成 UTF-8 字符串输出,浏览器或终端解码异常导致显示为方块、问号或空格。真正因编码转换出错的情况,通常发生在调用外部命令(如 exec('ipconfig') 或 shell_exec('ifconfig'))并直接 echo 结果时。
- Windows 下用
ipconfig命令返回 GBK 编码,PHP 默认按 UTF-8 输出就会显示乱码 - Linux 下
ifconfig一般为 UTF-8,但某些容器或精简系统可能 locale 不全,输出含不可见控制字符 - 不要对
$_SERVER数组做mb_convert_encoding()—— 它们是纯 ASCII 字符串,转码反而引入问题
Windows 下用 exec('ipconfig') 取 IP 出现中文乱码的修复方式
根本原因是 ipconfig 在中文 Windows 上默认输出 GBK 编码,而 PHP 脚本环境通常是 UTF-8。直接 echo 就会错位显示。
- 用
iconv('GBK', 'UTF-8//IGNORE', $output)转换,//IGNORE可跳过非法字节,避免警告 - 更稳妥:改用
chcp 65001 && ipconfig强制 cmd 输出 UTF-8(需确保系统支持),再用mb_convert_encoding($output, 'UTF-8', 'UTF-8')做无损透传 - 正则提取 IP 时,优先匹配 IPv4(
/\b\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}\b/),避开 IPv6 和中文字段干扰
跨平台安全取本机 IPv4 地址的推荐写法
绕开 shell 命令和编码问题,直接走 PHP 内置函数最可靠:
- 用
gethostbyname(gethostname())—— 简单,但只返回首个 A 记录,可能不是你期望的网卡地址(比如指向 127.0.0.1) - 用
net_get_interfaces()(PHP 7.4+)获取所有接口,过滤掉lo、docker0、veth*等虚拟网卡,再取第一个非空 IPv4:$interfaces = net_get_interfaces(); foreach ($interfaces as $name => $iface) { if (in_array($name, ['lo', 'docker0'])) continue; if (!empty($iface['ipv4'])) { $ip = $iface['ipv4'][0]; break; } } - 兼容旧版 PHP:遍历
gethostbynamel(gethostname())并用filter_var($addr, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4)校验
为什么别用 $_SERVER['REMOTE_ADDR'] 取本机 IP?
$_SERVER['REMOTE_ADDR'] 是客户端发起 HTTP 请求的真实 IP,不是服务器本机 IP。哪怕你在本地用 curl http://localhost,它也返回 127.0.0.1 或 ::1,而不是网卡上配的 192.168.x.x。混淆这个是导致“取到奇怪地址”的常见原因。
立即学习“PHP免费学习笔记(深入)”;
- 代理或 Nginx 转发后,
REMOTE_ADDR更是变成上一级代理的 IP,完全不可信 - 想获取服务监听的实际 IP,应查
$_SERVER['SERVER_ADDR'](Apache/Nginx 正常配置下是准确的) - 如果
SERVER_ADDR是::1,说明 Web 服务监听在 IPv6 localhost,可检查 Apache 的Listen [::]:80或 Nginx 的listen [::]:80配置
gethostname() 返回的是容器 ID,net_get_interfaces() 返回的可能是虚拟 bridge 地址而非宿主机真实出口 IP —— 这时候得结合 $_SERVER['SERVER_ADDR'] 或读取 /proc/net/route 才能准确定位。











