Hi,大家好,我是抢老婆酸奶的小肥仔。
最近在做公司需求时,甲方一会提出根据IP获取所在地,一会有提出根据手机号获取手机号所在地。真的是:需求时时变,累死程序猿。甲方才是爸爸。
之前已经实现过根据IP获取所在地的几种方式,大家可以参考我之前写的文章下:《SpringBoot通过ip获取归属地的几种方式》juejin.cn/post/728074...。
那么今天我们来看看根据手机号有哪些方式可以获取归属地呢?
废话不多说,开撸!
1、基于libphonenumber
libphonenumber:是谷歌提供的一款用于解析、格式化和校验国际手机号码的软件库。它提供了三个包,分别对应不同的功能。
libphonenumber
:用于校验手机号的正确性,提供了:getNumberType,isNumberMatch ,getExampleNumber 等方法。
carrier
:用于获取手机号的供应商。通过初始化PhoneNumberToCarrierMapper ,调用getNameForNumber可获取运营商信息。
geocoder
:用于获取手机号的归属地。通过初始化PhoneNumberOfflineGeocoder ,调用getDescriptionForNumber方法可获取手机归属地。
下面我们来说说具体实现。
引入包:
xml
<dependency>
<groupId>com.googlecode.libphonenumber</groupId>
<artifactId>libphonenumber</artifactId>
<version>8.13.26</version>
</dependency>
<dependency>
<groupId>com.googlecode.libphonenumber</groupId>
<artifactId>carrier</artifactId>
<version>1.210</version>
</dependency>
<dependency>
<groupId>com.googlecode.libphonenumber</groupId>
<artifactId>geocoder</artifactId>
<version>2.220</version>
</dependency>
1.1 编写工具
引入libphonenumber所有包后,我们编写一个工具类,实现手机校验,获取供应商,归属地等信息。
java
/**
* @author: jiangjs
* @description: 基于google的libphonenumber将手机号转成地区及供应商信息
* @date: 2023/11/30 14:33
**/
public class PhoneToRegionUtil {
/**
* 手机号基本工具类
*/
private final static PhoneNumberUtil PHONE_NUMBER_UTIL = PhoneNumberUtil.getInstance();
/**
* 运营商
*/
private final static PhoneNumberToCarrierMapper CARRIER_MAPPER = PhoneNumberToCarrierMapper.getInstance();
/**
*
*/
private final static PhoneNumberOfflineGeocoder GEO_CODER = PhoneNumberOfflineGeocoder.getInstance();
/**
* 验证当前手机号是否有效
* @param phone 手机号
* @return 校验结果
*/
public static boolean isValidNumber(String phone){
return PHONE_NUMBER_UTIL.isValidNumber(getPhoneNumber(phone));
}
/**
* 获取手机号运营商
* @param phone 手机号
* @return 运营商
*/
public static String getPhoneCarrier(String phone){
return isValidNumber(phone) ? CARRIER_MAPPER.getNameForNumber(getPhoneNumber(phone), Locale.CHINA) : "";
}
/**
* 获取手机号归属地
* @param phone 手机号
* @return 归属地
*/
public static String getRegionInfoByPhone(String phone){
return isValidNumber(phone) ? GEO_CODER.getDescriptionForNumber(getPhoneNumber(phone),Locale.CHINESE) : "";
}
/**
* 生成PhoneNumber
* @param phone 手机号
* @return PhoneNumber
*/
private static Phonenumber.PhoneNumber getPhoneNumber(String phone){
Phonenumber.PhoneNumber phoneNumber = new Phonenumber.PhoneNumber();
phoneNumber.setCountryCode(86);
phoneNumber.setNationalNumber(Long.parseLong(phone));
return phoneNumber;
}
/**
* 获取手机号的归属信息:运营商,归属地
* @param phone 手机号
* @return 归属信息
*/
public static JSONObject getPhoneAffiliationInfo(String phone){
JSONObject affiliation = new JSONObject();
affiliation.put("phone",phone);
affiliation.put("carrier",getPhoneCarrier(phone));
affiliation.put("region",getRegionInfoByPhone(phone));
return affiliation;
}
}
其中,getPhoneNumber创建每个手机号的Phonenumber.PhoneNumber,供其他接口调用。同时在调用运营商等接口时先进行手机号的校验。
在上面的接口中,我们会发现创建Phonenumber.PhoneNumber时,会使用setCountryCode方法去设置所在国家的电话区号,我们有时候复制手机号会发现前面是86,而86就是代表我们国家。每个国家有每个国家电话代号,其他国家代号,小伙伴们可以参考国际电信联盟根据 E.164 标准 分配给各国或特殊行政区的代码。
1.2 获取归属地
已经封装了工具类,那么接下来我们就测试一下,用手机号试试能不能获取归属信息。
我们直接在Controller层中编写接口:
java
@GetMapping("/getPhoneAffiliationInfo.do/{phone}")
public JsonResult<?> getPhoneAffiliationInfo(@PathVariable("phone") String phone){
return JsonResult.success(PhoneToRegionUtil.getPhoneAffiliationInfo(phone));
}
在浏览器中输入地址,添加号码:

通过测试,引用谷歌提供的包,可以解决我们的需求。
哈哈,可以不用加班.......
2、基于CSV文件
虽然引入谷歌的可以搞定需求了,但是作为程序员总要想想还有没有其他方式实现?这不又找一种方式。哈哈
其实我们的手机号是有规律可循的:
1、前3位:前三位的数字,其实代表的是运营商。不同的运营商会提供不同的号段。比如:我的手机号是135开头就是移动提供的。移动除了提供135号段外,还有其他各种号段,如134,137等;联通则提供了:130,131等号段;电信呢,提供了133,153等号段。
2、前7位:前7位则是可以确定手机号的归属地,例如:我的手机号前7位是1350154,则可以确定是广东省广州市。
既然我们知道了手机号的一些规律,那么如果有一份这样的文档,我们是不是就可以基于这份文档进行归属地的查询呢?
还真有这样的一份文档,我在网上找到一份4年前的CSV文档。如图:

既然有这份文档那我们就好实现了该功能。
2.1 读取CSV文件
只所以写读取CSV文件,是因为读取到这些信息后,想怎么查询就由我们自己说了算了。可以将数据存储到数据库查询,也可以放在redis中查询。下面我们基于redis的查询来实现归属功能。
将CSV文件读取到redis中。
java
/**
* @author: jiangjs
* @description: 服务启动后,加载数据到缓存
* @date: 2023/12/9 15:32
**/
@ConditionalOnProperty(havingValue = "true",value = "phoneToRegion.enabled")
@Component
public class ReadRegionToRedisStart implements CommandLineRunner {
private final static String REGION_CSV_PATH = "classpath:/static/region/phonetmp.csv";
private final static String PHONE_REGION_KEY = "country_phone_region_info";
@Resource
private ResourceLoader resourceLoader;
@Resource
private RedisTemplate<String,Object> redisTemplate;
@Override
public void run(String... args) {
long size = redisTemplate.opsForHash().size(PHONE_REGION_KEY);
if (size <= 0){
try (InputStream ism = resourceLoader.getResource(REGION_CSV_PATH).getInputStream()){
Assert.notNull(ism,"读取手机号信息文件为空");
BufferedReader reader = new BufferedReader(new InputStreamReader(ism));
String line = reader.readLine();
while (StringUtils.isNoneBlank(line)){
String[] lineVal = line.split(",");
RegionVo regionVo = new RegionVo();
regionVo.setPhonePrefix(lineVal[0]).setProvince(lineVal[1]).setCity(lineVal[2]).setCarrier(lineVal[3]);
redisTemplate.opsForHash().put(PHONE_REGION_KEY,lineVal[0],regionVo);
line = reader.readLine();
}
}catch (Exception e){
e.printStackTrace();
throw new RuntimeException("获取手机号信息报错");
}
}
}
}
我们在系统启动后,自动将CSV数据加载到redis中,当然通过@ConditionalOnProperty可以来自行决定要不要加载到内存中。不知道@ConditionalOnProperty注解使用的小伙伴去我的主页可以找到这篇文章来了解。
2.2 创建工具
数据被加载到内存后,那么我们就可以编写工具类来进行获取手机归属地。
java
/**
* @author: jiangjs
* @description: 读取CSV文件,根据手机号前7位进行匹配
* @date: 2023/11/30 14:54
**/
@Component
public class PhoneToRegionCsvUtil {
private final static String PHONE_REGION_KEY = "country_phone_region_info";
@Resource
private RedisTemplate<String,Object> redisTemplate;
/**
* 根据手机号获取手机归属地
* @param phone 手机号
* @return 归属地信息
*/
public RegionVo getPhoneToRegion(String phone){
String prefix = StringUtils.substring(phone, 0, 7);
Object region = redisTemplate.opsForHash().get(PHONE_REGION_KEY, prefix);
return Objects.isNull(region) ? new RegionVo() : (RegionVo) region;
}
}
2.3 获取归属地
有了工具类,那我们来测试一下。
直接在Controller层中编写接口:
java
@Resource
private PhoneToRegionCsvUtil phoneToRegionCsvUtil;
@GetMapping("/getPhoneGeoInfoByCsv.do/{phone}")
public JsonResult<?> getPhoneGeoInfoByCsv(@PathVariable("phone") String phone){
return JsonResult.success(phoneToRegionCsvUtil.getPhoneToRegion(phone));
}
浏览器中访问:

至此我们也可以正常的获取到手机号的归属地。
3、页面抓取
页面抓取这种方式,跟我之前的《SpringBoot通过ip获取归属地的几种方式》juejin.cn/post/728074...中的页面抓取方式是一样的,在这就不跟大家详细介绍了。
【总结】
文中介绍了三种方式进行手机号查询归属地的方式。
第一种:基于谷歌提供的国际解析包,引入后不用额外引入其他的东西,只需要写工具类即可,查询速度也比较快。
第二种:基于CSV文件的,不用额外引入具体的包,但是要引入CSV文件,大小在12M多,当然也可以将文件放在磁盘里,这样不用担心部署包过大。如果是基于内存查询的话,则需要依赖redis,增加了难度。
第三种:这个就不推荐了,毕竟依赖于第三方,如果服务挂了的话就没法使用了。如果用户量大的话,很可能会被第三方......,大家都懂的。
我在应用就是使用了第一种方式。
好了,今天就跟大家分享到这,谢谢大家。记得一定要给个赞,收藏再走哦.......