使用Java的request获取IP地址实例

本文向您介绍使用Java获取IP地址的方法,在request.getRemoteAddr()方法在使用方向代理失效的状况下,本文的方法依然能够获取真是地址。  浏览器


在JSP里,获取客户端的IP地址的方法是:request.getRemoteAddr(),这种方法在大部分状况下都是有效的。可是在经过了Apache,Squid等反向代理软件就不能获取到客户端的真实IP地址了。 服务器

若是使用了反向代理软件,将http://192.168.1.110:2046/ 的URL反向代理为 http://www.bt285.cn / 的URL时,用request.getRemoteAddr()方法Java获取的IP地址是:127.0.0.1 或 192.168.1.110,而并非客户端的真实IP。 jsp

通过代理之后,因为在客户端和服务之间增长了中间层,所以服务器没法直接拿到客户端的IP,服务器端应用也没法直接经过转发请求的地址返回给客户端。可是在转发请求的HTTP头信息中,增长了X-FORWARDED-FOR信息。用以跟踪原有的客户端IP地址和原来客户端请求的服务器地址。当咱们访问http://www.5q520.cn /index.jsp/ 时,其实并非咱们浏览器真正访问到了服务器上的index.jsp文件,而是先由代理服务器去访问http://192.168.1.110:2046/index.jsp ,代理服务器再将访问到的结果返回给咱们的浏览器,由于是代理服务器去访问index.jsp的,因此index.jsp中经过request.getRemoteAddr()的方法获取的IP其实是代理服务器的地址,并非客户端的IP地址。 oop

因而可得出Java获取IP地址真实地址的方法一: 测试

   
  1. public String getRemortIP(HttpServletRequest request) {      
  2.   if (request.getHeader("x-forwarded-for") == null) {      
  3.    return request.getRemoteAddr();      
  4.   }      
  5.   return request.getHeader("x-forwarded-for");      
  6.  }   

但是当我访问http://www.5a520.cn /index.jsp/ 时,返回的IP地址始终是unknown,也并非如上所示的127.0.0.1 或 192.168.1.110了,而我访问http://192.168.1.110:2046/index.jsp 时,则能返回客户端的真实IP地址,写了个方法去验证。缘由出在了Squid上。squid.conf 的配制文件 forwarded_for 项默认是为on,若是 forwarded_for 设成了 off  则:X-Forwarded-For: unknown ui

因而可得出得到Java获取IP地址的方法二: spa

   
  1. public String getIpAddr(HttpServletRequest request) {      
  2.        String ip = request.getHeader("x-forwarded-for");      
  3.        if(ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {      
  4.            ip = request.getHeader("Proxy-Client-IP");      
  5.        }      
  6.        if(ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {      
  7.            ip = request.getHeader("WL-Proxy-Client-IP");      
  8.        }      
  9.        if(ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {      
  10.            ip = request.getRemoteAddr();      
  11.        }      
  12.        return ip;      
  13.    }   

但是,若是经过了多级反向代理的话,X-Forwarded-For的值并不止一个,而是一串IP值,究竟哪一个才是真正的用户端的真实IP呢? 代理

答案是取X-Forwarded-For中第一个非unknown的有效IP字符串。 code

如: ip

X-Forwarded-For:192.168.1.110, 192.168.1.120, 192.168.1.130, 192.168.1.100

用户真实IP为: 192.168.1.110

经过这样的测试


还有一些其余的方法能够实现获取ip:以下

方法一(只能在Windows上使用,Linux平台就gei屁了):

try
{
System.out.println("本机的IP = " + InetAddress.getLocalHost());
} catch (UnknownHostException e)
{
e.printStackTrace();
}

在Linux下的执行结果是:本机的IP = xxx/127.0.1.1 (其中xxx是你的计算机名,偶这里马赛克了)

方法二(宣称能够在Linux下执行)

Enumeration netInterfaces=NetworkInterface.getNetworkInterfaces();
InetAddress ip = null;
while(netInterfaces.hasMoreElements())
{
NetworkInterface ni=(NetworkInterface)netInterfaces.nextElement();
System.out.println(ni.getName());
ip=(InetAddress) ni.getInetAddresses().nextElement();
if( !ip.isSiteLocalAddress()
&& !ip.isLoopbackAddress()
&& ip.getHostAddress().indexOf(":")==-1)
{
System.out.println ("本机的ip=" + ip.getHostAddress());
break;
}
else
{
ip=null;
}
}
从红色部分的代码能够看到,该代码对于获取到的第一个NetworkInterface的IP地址的获取,没有循环的获取,只是对第一个IP地址进行了处理,这样就致使了若是第一个IP地址不是一个
Inet4Address的地址而是一个< span
id="ArticleContent1_ArticleContent1_lblContent">Inet6Address,这个判断 ip.getHostAddress().indexOf(":")==-1将永远是false,这个if条件进不去呀,多害人,强烈鄙视!

不过方法二思路是对了,就是有些小毛病,让偶修改了一下,最终版的能够在 Linux下正确执行的代码以下:
Enumeration allNetInterfaces = NetworkInterface.getNetworkInterfaces();
InetAddress ip = null;
while (allNetInterfaces.hasMoreElements())
{
NetworkInterface netInterface = (NetworkInterface) allNetInterfaces.nextElement();
System.out.println(netInterface.getName());
Enumeration addresses = netInterface.getInetAddresses();
while (addresses.hasMoreElements())
{
ip = (InetAddress) addresses.nextElement();
if (ip != null && ip instanceof Inet4Address)
{
System.out.println("本机的IP = " + ip.getHostAddress());
}
}
}

 

还有一种获取方式:

InetAddress address = InetAddress.getLocalHost();

String ip = address.getHostAddress();

相关文章
相关标签/搜索