获取终端外网IP地址

  • 服务端提供获取终端外网IP地址的IP,http接口的形式
  • 以下只提供获取到 IPV4 的地址
  • 把以下接口部署到服务器中,客户端请求访问服务端时获取到当前客户端的外网IP地址
  • 如果部署到本地,访问时返回的为 null
java 复制代码
/**
 * ClientIpController
 *
 * @author xdr630
 * @since 2025/7/6
 */
@RestController
public class ClientIpController {

    private static final Logger LOGGER = LoggerFactory.getLogger(ClientIpController .class);

    @GetMapping("/getRemoteIp")
    @ResponseBody
    public String getClientIp(HttpServletRequest request) {

        String clientIp = null;

        try {
            String xForwardedFor = request.getHeader("X-Forwarded-For");
            if (xForwardedFor != null && !xForwardedFor.isEmpty() && !"unknown".equalsIgnoreCase(xForwardedFor)) {
                // X-Forwarded-For:Squid 服务代理,X-Forwarded-For 可能包含多个 IP,取第一个非空的 IP
                clientIp = xForwardedFor.split(",")[0].trim();
            }

            if (clientIp == null || clientIp.isEmpty() || "unknown".equalsIgnoreCase(clientIp)) {
                // Proxy-Client-IP:apache 服务代理
                clientIp = request.getHeader("Proxy-Client-IP");
            }

            if (clientIp == null || clientIp.isEmpty() || "unknown".equalsIgnoreCase(clientIp)) {
                // WL-Proxy-Client-IP:weblogic 服务代理
                clientIp = request.getHeader("WL-Proxy-Client-IP");
            }

            if ((clientIp == null || clientIp.isEmpty() || "unknown".equalsIgnoreCase(clientIp))) {
                // X-Real-IP:nginx服务代理
                clientIp = request.getHeader("X-Real-IP");
            }

            if (clientIp == null || clientIp.isEmpty() || "unknown".equalsIgnoreCase(clientIp)) {
                clientIp = request.getRemoteAddr();
            }

            if (clientIp != null && !clientIp.isEmpty()) {
                clientIp = clientIp.split(",")[0];
            }

            if (!isValidIPv4(clientIp)) {
                LOGGER.warn("无效的客户端 IP: {}", clientIp);
                // 在浏览器端显示返回null
                clientIp = "null";
            }
        } catch (Exception e) {
            LOGGER.error("获取客户端公网IP失败", e);
            throw new RuntimeException(e);
        }

        return clientIp;
    }

    private boolean isValidIPv4(String ip) {
        String ipv4Regex = "^((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$";
        return Pattern.compile(ipv4Regex).matcher(ip).matches();
    }
}
  • clientIp = "null"; 如果去掉这段的话,当本地访问这个接口时,返回的地址就会为如下,这里改为字符串null更好点

IPv6 地址 0:0:0:0:0:0:0:1 是一个特殊的保留地址,它等价于 IPv4 中的 127.0.0.1。这个地址表示本机回环地址(Loopback Address)

含义:

  • IPv6 回环地址::1
    • 这个地址通常被简写为 ::1,而不是写全八个段 0:0:0:0:0:0:0:1
    • 它用于代表本机(localhost),也就是说,当你在一台设备上使用这个地址时,网络通信不会通过物理网络接口,而是直接返回到本机。

用途:

  • 测试本地网络应用(如 Web 服务器、数据库等)。
  • 配置服务监听时,绑定到 ::1 表示只接受本机的连接请求。
  • 开发和调试网络程序时非常有用。

类比:

地址类型 地址 说明
IPv4 127.0.0.1 本机回环地址
IPv6 ::1 IPv6 的本机回环地址

看到的 0:0:0:0:0:0:0:1 就是 ::1 的完整写法,代表本机地址。

相关推荐
Cherry的跨界思维5 分钟前
28、AI测试环境搭建与全栈工具实战:从本地到云平台的完整指南
java·人工智能·vue3·ai测试·ai全栈·测试全栈·ai测试全栈
alonewolf_9943 分钟前
JDK17新特性全面解析:从语法革新到模块化革命
java·开发语言·jvm·jdk
一嘴一个橘子44 分钟前
spring-aop 的 基础使用(啥是增强类、切点、切面)- 2
java
sheji34161 小时前
【开题答辩全过程】以 中医药文化科普系统为例,包含答辩的问题和答案
java
sunfove1 小时前
光网络的立交桥:光开关 (Optical Switch) 原理与主流技术解析
网络
恋爱绝缘体11 小时前
2020重学C++重构你的C++知识体系
java·开发语言·c++·算法·junit
wszy18092 小时前
新文章标签:让用户一眼发现最新内容
java·python·harmonyos
wszy18092 小时前
顶部标题栏的设计与实现:让用户知道自己在哪
java·python·react native·harmonyos
程序员小假3 小时前
我们来说一下无锁队列 Disruptor 的原理
java·后端
资生算法程序员_畅想家_剑魔3 小时前
Kotlin常见技术分享-02-相对于Java 的核心优势-协程
java·开发语言·kotlin