likes
comments
collection
share

SpringBoot通过手机号获取归属地的几种方式。

作者站长头像
站长
· 阅读数 8

Hi,大家好,我是抢老婆酸奶的小肥仔。

最近在做公司需求时,甲方一会提出根据IP获取所在地,一会又提出根据手机号获取手机号归属地。真的是:需求时时变,累死程序猿。甲方才是爸爸。SpringBoot通过手机号获取归属地的几种方式。

那么今天我们来看看根据手机号有哪些方式可以获取归属地呢?

废话不多说,开撸!

1、基于libphonenumber

libphonenumber:是谷歌提供的一款用于解析、格式化和校验国际手机号码的软件库。它提供了三个包,分别对应不同的功能。

libphonenumber:用于校验手机号的正确性,提供了:getNumberType,isNumberMatch ,getExampleNumber 等方法。

carrier:用于获取手机号的供应商。通过初始化PhoneNumberToCarrierMapper ,调用getNameForNumber可获取运营商信息。

geocoder:用于获取手机号的归属地。通过初始化PhoneNumberOfflineGeocoder ,调用getDescriptionForNumber方法可获取手机归属地。

下面我们来说说具体实现。

引入包:

 <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所有包后,我们编写一个工具类,实现手机校验,获取供应商,归属地等信息。

/**
 * @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层中编写接口:

@GetMapping("/getPhoneAffiliationInfo.do/{phone}")
public JsonResult<?> getPhoneAffiliationInfo(@PathVariable("phone") String phone){
    return JsonResult.success(PhoneToRegionUtil.getPhoneAffiliationInfo(phone));
}

在浏览器中输入地址,添加号码:

SpringBoot通过手机号获取归属地的几种方式。

通过测试,引用谷歌提供的包,可以解决我们的需求。

哈哈,可以不用加班.......

2、基于CSV文件

虽然引入谷歌的可以搞定需求了,但是作为程序员总要想想还有没有其他方式实现?这不又找一种方式。哈哈

其实我们的手机号是有规律可循的:

1、前3位:前三位的数字,其实代表的是运营商。不同的运营商会提供不同的号段。比如:我的手机号是135开头就是移动提供的。移动除了提供135号段外,还有其他各种号段,如134,137等;联通则提供了:130,131等号段;电信呢,提供了133,153等号段。

2、前7位:前7位则是可以确定手机号的归属地,例如:我的手机号前7位是1350154,则可以确定是广东省广州市。

既然我们知道了手机号的一些规律,那么如果有一份这样的文档,我们是不是就可以基于这份文档进行归属地的查询呢?

还真有这样的一份文档,我在网上找到一份4年前的CSV文档。如图:

SpringBoot通过手机号获取归属地的几种方式。

既然有这份文档那我们就好实现了该功能。

2.1 读取CSV文件

只所以写读取CSV文件,是因为读取到这些信息后,想怎么查询就由我们自己说了算了。可以将数据存储到数据库查询,也可以放在redis中查询。下面我们基于redis的查询来实现归属功能。

将CSV文件读取到redis中。

/**
 * @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 创建工具

数据被加载到内存后,那么我们就可以编写工具类来进行获取手机归属地。

/**
 * @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层中编写接口:

@Resource
private PhoneToRegionCsvUtil phoneToRegionCsvUtil;

@GetMapping("/getPhoneGeoInfoByCsv.do/{phone}")
public JsonResult<?> getPhoneGeoInfoByCsv(@PathVariable("phone") String phone){
    return JsonResult.success(phoneToRegionCsvUtil.getPhoneToRegion(phone));
}

浏览器中访问:

SpringBoot通过手机号获取归属地的几种方式。

至此我们也可以正常的获取到手机号的归属地。

3、页面抓取

【总结】

文中介绍了三种方式进行手机号查询归属地的方式。

第一种:基于谷歌提供的国际解析包,引入后不用额外引入其他的东西,只需要写工具类即可,查询速度也比较快。

第二种:基于CSV文件的,不用额外引入具体的包,但是要引入CSV文件,大小在12M多,当然也可以将文件放在磁盘里,这样不用担心部署包过大。如果是基于内存查询的话,则需要依赖redis,增加了难度。

第三种:这个就不推荐了,毕竟依赖于第三方,如果服务挂了的话就没法使用了。如果用户量大的话,很可能会被第三方......,大家都懂的。

我在应用就是使用了第一种方式。

好了,今天就跟大家分享到这,谢谢大家。记得一定要给个赞,收藏再走哦.......

码云地址:gitee.com/lovequeena/…