在Servlet中获取真实的客户端IP

时间:2013-04-23 07:52:45

标签: java servlets ip clientip

我遇到一个简单问题的麻烦。我会在HTTPServlet中获得真实客户端IP

从现在起我用过:

request.getRemoteAddr()

但现在它返回一个虚假的IP。例如:xxx.xxx.xxx。 50 但我的IP类似于xxx.xxx.xxx。 159 。 (在http://whatismyipaddress.com/处查看)。

现在我尝试使用:

request.getHeader("X-Forwarded-For")

返回NULL。

我还参加了以下课程的探测:

public class IpUtils {

public static final String _255 = "(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)";
public static final Pattern pattern = Pattern.compile("^(?:" + _255 + "\\.){3}" + _255 + "$");

public static String longToIpV4(long longIp) {
    int octet3 = (int) ((longIp >> 24) % 256);
    int octet2 = (int) ((longIp >> 16) % 256);
    int octet1 = (int) ((longIp >> 8) % 256);
    int octet0 = (int) ((longIp) % 256);

    return octet3 + "." + octet2 + "." + octet1 + "." + octet0;
}

public static long ipV4ToLong(String ip) {
    String[] octets = ip.split("\\.");
    return (Long.parseLong(octets[0]) << 24) + (Integer.parseInt(octets[1]) << 16)
            + (Integer.parseInt(octets[2]) << 8) + Integer.parseInt(octets[3]);
}

public static boolean isIPv4Private(String ip) {
    long longIp = ipV4ToLong(ip);
    return (longIp >= ipV4ToLong("10.0.0.0") && longIp <= ipV4ToLong("10.255.255.255"))
            || (longIp >= ipV4ToLong("172.16.0.0") && longIp <= ipV4ToLong("172.31.255.255"))
            || longIp >= ipV4ToLong("192.168.0.0") && longIp <= ipV4ToLong("192.168.255.255");
}

public static boolean isIPv4Valid(String ip) {
    return pattern.matcher(ip).matches();
}

public static String getIpFromRequest(HttpServletRequest request) {
    String ip;
    boolean found = false;
    if ((ip = request.getHeader("x-forwarded-for")) != null) {
        StringTokenizer tokenizer = new StringTokenizer(ip, ",");
        while (tokenizer.hasMoreTokens()) {
            ip = tokenizer.nextToken().trim();
            if (isIPv4Valid(ip) && !isIPv4Private(ip)) {
                found = true;
                break;
            }
        }
    }

    if (!found) {
        ip = request.getRemoteAddr();
    }

    return ip;
}
}

它还返回了xxx.xxx.xxx.50 IP。 :(

现在我不知道如何获得真正的客户端IP。如果有人知道解决方案,请回答。

3 个答案:

答案 0 :(得分:31)

我认为您的问题是您在本地网络中的某个位置运行服务器,因此您可以在该网络中获取IP。但是,当您尝试使用发现IP地址的在线服务时,您的IP是服务提供商的路由器的IP或类似的东西。使用request.getRemoteAddr()是正确的。这就是这些服务的作用,而且没有其他设施。

答案 1 :(得分:12)

在servlet容器之前有NAT(网络地址转换)设备时 request.getRemoteAddr()返回NAT的地址,因为它启动了新连接。

要解决您的问题,您应该配置NAT以在标头中发送原始客户端的IP。 然后,从请求中获取标头。如果您在NAT X-Forwarded-For标头中配置,请使用以下代码:request.getHeader("X-Forwarded-For")

您的功能是正确的,您可以在配置NAT时使用它。

答案 2 :(得分:10)

<强>爪哇

String ip = request.getRemoteAddr();
if (ip.equalsIgnoreCase("0:0:0:0:0:0:0:1")) {
    InetAddress inetAddress = InetAddress.getLocalHost();
    String ipAddress = inetAddress.getHostAddress();
    ip = ipAddress;
}
model.addAttribute("IP", ip);

JSP <span><B>IP : <c:out value="${IP}" /></B></span> // Spring MAV

<强> JSP

<%@ page language="java" contentType="text/html; charset=ISO-8859-1"   pageEncoding="ISO-8859-1"%>
<%@ page import="java.net.InetAddress" %>

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
<title>Displays Client IP Address</title>
</head>
<body>
<%
    // paste above java code 
out.println("<br><B>Clients IP : </B>"+ip);
%>
<span><br>Local Host  : <%= request.getRemoteAddr() %></span>
<span><br>ServerPort  : <%= request.getServerPort() %></span>
<span><br>Path        : <%= request.getRequestURI()%></span>
</span>
</body>
</html>

如果jsp部署在本地服务器上并且访问该jsp,则返回0:0:0:0:0:0:0:1。 如果使用ip在本地访问应用程序,则返回IP。

RESTful带泽西岛的网络服务

@javax.ws.rs.Path("/ip")
public class GetIPInfo {

    @GET    @Produces(MediaType.TEXT_PLAIN) 
    public Response getIP(@Context javax.servlet.http.HttpServletRequest request) throws UnknownHostException { 

        String remoteHost = request.getRemoteHost();
        String remoteAddr = request.getRemoteAddr();
        if (remoteAddr.equals("0:0:0:0:0:0:0:1")) {
            InetAddress localip = java.net.InetAddress.getLocalHost();
            remoteAddr = localip.getHostAddress();
            remoteHost = localip.getHostName();
        }
        int remotePort = request.getRemotePort();

        String msgoutput = remoteHost + " (" + remoteAddr + " : " + remotePort + ")";
        return Response.status(200).entity(msgoutput).build(); 
    }
}

使用JavaScript查找 IP