前言
最近的一个项目中需要将不同省份的用户,展示不同内容,所以需要通过Request请求获取IP地址, 然后通过IP获取IP对应省份。
这里的操作步骤一共有步:
1. 通过Request获取IP
2. 通过IP获取对应省份、城市
3. 通过设置的省份和IP对应省份进行比对,展示内容
通过Request获取IP
可以参考我的另外一篇文章【Java 通过Request请求获取IP地址】下面是代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
public class IpAdrressUtil {
/**
* 获取Ip地址
* @param request
* @return
*/
private static String getIpAdrress(HttpServletRequest request) {
String Xip = request.getHeader( "X-Real-IP" );
String XFor = request.getHeader( "X-Forwarded-For" );
if (StringUtils.isNotEmpty(XFor) && ! "unKnown" .equalsIgnoreCase(XFor)){
//多次反向代理后会有多个ip值,第一个ip才是真实ip
int index = XFor.indexOf( "," );
if (index != - 1 ){
return XFor.substring( 0 ,index);
} else {
return XFor;
}
}
XFor = Xip;
if (StringUtils.isNotEmpty(XFor) && ! "unKnown" .equalsIgnoreCase(XFor)){
return XFor;
}
if (StringUtils.isBlank(XFor) || "unknown" .equalsIgnoreCase(XFor)) {
XFor = request.getHeader( "Proxy-Client-IP" );
}
if (StringUtils.isBlank(XFor) || "unknown" .equalsIgnoreCase(XFor)) {
XFor = request.getHeader( "WL-Proxy-Client-IP" );
}
if (StringUtils.isBlank(XFor) || "unknown" .equalsIgnoreCase(XFor)) {
XFor = request.getHeader( "HTTP_CLIENT_IP" );
}
if (StringUtils.isBlank(XFor) || "unknown" .equalsIgnoreCase(XFor)) {
XFor = request.getHeader( "HTTP_X_FORWARDED_FOR" );
}
if (StringUtils.isBlank(XFor) || "unknown" .equalsIgnoreCase(XFor)) {
XFor = request.getRemoteAddr();
}
return XFor;
}
}
|
通过IP获取对应省份、城市
使用【GeoLite2 City】库
目前开源的IP地址库与城市对应关系用的比较多的是MaxMind公司的GeoLite数据库,GeoLite数据库有开源版本和收费版本,我们使用的是开源版本,GeoLite目前已经更新到2了,所以我们下载GeoLite2 City库。
官方下载地址是【http://dev.maxmind.com/geoip/geoip2/geolite2/】
如果觉得慢就用迅雷下。下载完成后就是,下载完成就解压。得到【GeoLite2-City.mmdb】文件,这个就是数据库。
Java例子是这样使用的:
首先在项目中加入maven支持
1
2
3
4
5
|
<dependency>
<groupId>com.maxmind.geoip2</groupId>
<artifactId>geoip2</artifactId>
<version> 2.8 . 1 </version>
</dependency>
|
然后通过GeoLite2查询得到省份、城市:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
public static void main(String[] args) throws Exception{
// 创建 GeoLite2 数据库
File database = new File( "/Users/admin/GeoLite2-City.mmdb" );
// 读取数据库内容
DatabaseReader reader = new DatabaseReader.Builder(database).build();
InetAddress ipAddress = InetAddress.getByName( "171.108.233.157" );
// 获取查询结果
CityResponse response = reader.city(ipAddress);
// 获取国家信息
Country country = response.getCountry();
System.out.println(country.getIsoCode()); // 'CN'
System.out.println(country.getName()); // 'China'
System.out.println(country.getNames().get( "zh-CN" )); // '中国'
// 获取省份
Subdivision subdivision = response.getMostSpecificSubdivision();
System.out.println(subdivision.getName()); // 'Guangxi Zhuangzu Zizhiqu'
System.out.println(subdivision.getIsoCode()); // '45'
System.out.println(subdivision.getNames().get( "zh-CN" )); // '广西壮族自治区'
// 获取城市
City city = response.getCity();
System.out.println(city.getName()); // 'Nanning'
Postal postal = response.getPostal();
System.out.println(postal.getCode()); // 'null'
System.out.println(city.getNames().get( "zh-CN" )); // '南宁'
Location location = response.getLocation();
System.out.println(location.getLatitude()); // 22.8167
System.out.println(location.getLongitude()); // 108.3167
}
|
如果是生产环境,可以直接创建一个Service,在Service初始化的时候创建reader对象,然后在公共方法中通过ip查询地址,下面以省份为例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
|
import com.maxmind.geoip2.DatabaseReader;
import com.maxmind.geoip2.model.CityResponse;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.env.Environment;
import org.springframework.stereotype.Service;
import javax.annotation.PostConstruct;
import java.io.File;
import java.net.InetAddress;
/**
* IP地址服务
*/
@Service
public class IpAddressService {
private static Logger logger = LoggerFactory.getLogger(IpAddressService. class );
private static String dbPath = "/usr/local/GeoLite2-City.mmdb" ;
private static DatabaseReader reader;
@Autowired
private Environment env;
@PostConstruct
public void init() {
try {
String path = env.getProperty( "geolite2.city.db.path" );
if (StringUtils.isNotBlank(path)) {
dbPath = path;
}
File database = new File(dbPath);
reader = new DatabaseReader.Builder(database).build();
} catch (Exception e) {
logger.error( "IP地址服务初始化异常:" + e.getMessage(), e);
}
}
public String getSubdivision(String ipAddress){
try {
CityResponse response = reader.city(InetAddress.getByName(ipAddress));
return response.getMostSpecificSubdivision().getNames().get( "zh-CN" );
} catch (Exception e){
logger.error( "根据IP[{}]获取省份失败:{}" , ipAddress, e.getMessage());
return null ;
}
}
}
|
然后在需要的地方就行判断:
1
2
3
4
5
6
|
String areaNames = { "北京" , "天津" , "上海" };
String subdivision = ipAddressService.getSubdivision(getIpAdrress(request));
// 匹配
if (containsArea(subdivision, areaNames)){
// 处理...
}
|
匹配方法:
1
2
3
4
5
6
7
8
9
10
11
|
private boolean containsArea(String name, String[] areaNames) {
if (StringUtils.isBlank(name)){
return false ;
}
for (String areaName : areaNames){
if (name.contains(areaName)){
return true ;
}
}
return false ; // 按地域返回数据
}
|
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:http://blog.csdn.net/chwshuang/article/details/71951000