1. 程式人生 > >如何通過Request獲取使用者真實IP

如何通過Request獲取使用者真實IP

在Servlet裡,獲取客戶端的IP地址的方法是:request.getRemoteAddr(),這種方法在大部分情況下都是有效的。但是在通過了Apache,Squid,Nginx等反向代理軟體就不能獲取到客戶端的真實IP地址了。

如果使用了反向代理軟體,例如將http://192.168.101.88:80/ 的URL反向代理為http://pay.kedou.com/ 的URL時,用request.getRemoteAddr()方法獲取的IP地址是:127.0.0.1 或192.168.101.88,而並不是客戶端的真實IP。

如下圖,原來是client端直接請求服務端,走A路線請求,這時候通過request.getRemoteAddr()方法可以準備的獲取客戶端的IP。但是做了代理之後呢,client端不是直接請求服務端,而是走B線路請求代理伺服器,由代理器去請求服務端,這時候服務端通過request.getRemoteAddr()方法拿到的理所當然是代理伺服器的地址了。

經過代理以後,由於在客戶端和服務之間增加了中間層,因此伺服器無法直接拿到客戶端的IP,伺服器端應用也無法直接通過轉發請求的地址返回給客戶端。但是在轉發請求的HTTP頭資訊中,增加了X-FORWARDED-FOR資訊。用以跟蹤原有的客戶端IP地址和原來客戶端請求的伺服器地址。當我們訪問http://www.xxx.com/index.jsp/ 時,其實並不是我們瀏覽器真正訪問到了伺服器上的index.jsp檔案,而是先由代理伺服器去訪問http://192.168.1.110:2046/index.jsp ,代理伺服器再將訪問到的結果返回給我們的瀏覽器,因為是代理伺服器去訪問index.jsp的,所以index.jsp中通過request.getRemoteAddr()的方法獲取的IP實際上是代理伺服器的地址,並不是客戶端的IP地址。


下面的獲取真實IP的Java類:
package com.wyy.snail.user.utils;  
 
import java.io.IOException;  
 
import javax.servlet.http.HttpServletRequest;  
 
import org.apache.log4j.Logger;  
 
import com.wyy.snail.core.util.RandomCodeUtils;  
 
/**
 * 常用獲取客戶端資訊的工具
 * @author  dongyang - gong
 */  
public final class NetworkUtil {  
    /**
     * Logger for this class
     */  
    private static Logger logger = Logger.getLogger(RandomCodeUtils.class);  
 
    /**
     * 獲取請求主機IP地址,如果通過代理進來,則透過防火牆獲取真實IP地址;
     *  
     * @param request
     * @return
     * @throws IOException
     */  
    public final static String getIpAddress(HttpServletRequest request) throws IOException {  
        // 獲取請求主機IP地址,如果通過代理進來,則透過防火牆獲取真實IP地址  
 
        String ip = request.getHeader("X-Forwarded-For");  
        if (logger.isInfoEnabled()) {  
            logger.info("getIpAddress(HttpServletRequest) - X-Forwarded-For - String ip=" + ip);  
        }  
 
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {  
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {  
                ip = request.getHeader("Proxy-Client-IP");  
                if (logger.isInfoEnabled()) {  
                    logger.info("getIpAddress(HttpServletRequest) - Proxy-Client-IP - String ip=" + ip);  
                }  
            }  
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {  
                ip = request.getHeader("WL-Proxy-Client-IP");  
                if (logger.isInfoEnabled()) {  
                    logger.info("getIpAddress(HttpServletRequest) - WL-Proxy-Client-IP - String ip=" + ip);  
                }  
            }  
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {  
                ip = request.getHeader("HTTP_CLIENT_IP");  
                if (logger.isInfoEnabled()) {  
                    logger.info("getIpAddress(HttpServletRequest) - HTTP_CLIENT_IP - String ip=" + ip);  
                }  
            }  
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {  
                ip = request.getHeader("HTTP_X_FORWARDED_FOR");  
                if (logger.isInfoEnabled()) {  
                    logger.info("getIpAddress(HttpServletRequest) - HTTP_X_FORWARDED_FOR - String ip=" + ip);  
                }  
            }  
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {  
                ip = request.getRemoteAddr();  
                if (logger.isInfoEnabled()) {  
                    logger.info("getIpAddress(HttpServletRequest) - getRemoteAddr - String ip=" + ip);  
                }  
            }  
        } else if (ip.length() > 15) {  
            String[] ips = ip.split(",");  
            for (int index = 0; index < ips.length; index++) {  
                String strIp = (String) ips[index];  
                if (!("unknown".equalsIgnoreCase(strIp))) {  
                    ip = strIp;  
                    break;  
                }  
            }  
        }  
        return ip;  
    }  
}