SpringBoot實(shí)現(xiàn)根據(jù)手機(jī)號(hào)獲取歸屬地
最近在做公司需求時(shí),甲方一會(huì)提出根據(jù)IP獲取所在地,一會(huì)有提出根據(jù)手機(jī)號(hào)獲取手機(jī)號(hào)所在地。真的是:需求時(shí)時(shí)變,累死程序猿。甲方才是爸爸。
之前已經(jīng)實(shí)現(xiàn)過(guò)根據(jù)IP獲取所在地的幾種方式,大家可以參考我之前寫(xiě)的文章下:《SpringBoot通過(guò)ip獲取歸屬地的幾種方式》。
那么今天我們來(lái)看看根據(jù)手機(jī)號(hào)有哪些方式可以獲取歸屬地呢?
廢話不多說(shuō),開(kāi)擼!
1、基于libphonenumber
libphonenumber:是谷歌提供的一款用于解析、格式化和校驗(yàn)國(guó)際手機(jī)號(hào)碼的軟件庫(kù)。它提供了三個(gè)包,分別對(duì)應(yīng)不同的功能。
libphonenumber
:用于校驗(yàn)手機(jī)號(hào)的正確性,提供了:getNumberType,isNumberMatch ,getExampleNumber 等方法。
carrier
:用于獲取手機(jī)號(hào)的供應(yīng)商。通過(guò)初始化PhoneNumberToCarrierMapper ,調(diào)用getNameForNumber可獲取運(yùn)營(yíng)商信息。
geocoder
:用于獲取手機(jī)號(hào)的歸屬地。通過(guò)初始化PhoneNumberOfflineGeocoder ,調(diào)用getDescriptionForNumber方法可獲取手機(jī)歸屬地。
下面我們來(lái)說(shuō)說(shuō)具體實(shí)現(xiàn)。
引入包:
<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 編寫(xiě)工具
引入libphonenumber所有包后,我們編寫(xiě)一個(gè)工具類,實(shí)現(xiàn)手機(jī)校驗(yàn),獲取供應(yīng)商,歸屬地等信息。
/** * @author: jiangjs * @description: 基于google的libphonenumber將手機(jī)號(hào)轉(zhuǎn)成地區(qū)及供應(yīng)商信息 * @date: 2023/11/30 14:33 **/ public class PhoneToRegionUtil { /** * 手機(jī)號(hào)基本工具類 */ private final static PhoneNumberUtil PHONE_NUMBER_UTIL = PhoneNumberUtil.getInstance(); /** * 運(yùn)營(yíng)商 */ private final static PhoneNumberToCarrierMapper CARRIER_MAPPER = PhoneNumberToCarrierMapper.getInstance(); /** * */ private final static PhoneNumberOfflineGeocoder GEO_CODER = PhoneNumberOfflineGeocoder.getInstance(); /** * 驗(yàn)證當(dāng)前手機(jī)號(hào)是否有效 * @param phone 手機(jī)號(hào) * @return 校驗(yàn)結(jié)果 */ public static boolean isValidNumber(String phone){ return PHONE_NUMBER_UTIL.isValidNumber(getPhoneNumber(phone)); } /** * 獲取手機(jī)號(hào)運(yùn)營(yíng)商 * @param phone 手機(jī)號(hào) * @return 運(yùn)營(yíng)商 */ public static String getPhoneCarrier(String phone){ return isValidNumber(phone) ? CARRIER_MAPPER.getNameForNumber(getPhoneNumber(phone), Locale.CHINA) : ""; } /** * 獲取手機(jī)號(hào)歸屬地 * @param phone 手機(jī)號(hào) * @return 歸屬地 */ public static String getRegionInfoByPhone(String phone){ return isValidNumber(phone) ? GEO_CODER.getDescriptionForNumber(getPhoneNumber(phone),Locale.CHINESE) : ""; } /** * 生成PhoneNumber * @param phone 手機(jī)號(hào) * @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; } /** * 獲取手機(jī)號(hào)的歸屬信息:運(yùn)營(yíng)商,歸屬地 * @param phone 手機(jī)號(hào) * @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創(chuàng)建每個(gè)手機(jī)號(hào)的Phonenumber.PhoneNumber,供其他接口調(diào)用。同時(shí)在調(diào)用運(yùn)營(yíng)商等接口時(shí)先進(jìn)行手機(jī)號(hào)的校驗(yàn)。
在上面的接口中,我們會(huì)發(fā)現(xiàn)創(chuàng)建Phonenumber.PhoneNumber時(shí),會(huì)使用setCountryCode方法去設(shè)置所在國(guó)家的電話區(qū)號(hào),我們有時(shí)候復(fù)制手機(jī)號(hào)會(huì)發(fā)現(xiàn)前面是86,而86就是代表我們國(guó)家。每個(gè)國(guó)家有每個(gè)國(guó)家電話代號(hào),其他國(guó)家代號(hào),小伙伴們可以參考國(guó)際電信聯(lián)盟根據(jù) E.164 標(biāo)準(zhǔn) 分配給各國(guó)或特殊行政區(qū)的代碼。
1.2 獲取歸屬地
已經(jīng)封裝了工具類,那么接下來(lái)我們就測(cè)試一下,用手機(jī)號(hào)試試能不能獲取歸屬信息。
我們直接在Controller層中編寫(xiě)接口:
@GetMapping("/getPhoneAffiliationInfo.do/{phone}") public JsonResult<?> getPhoneAffiliationInfo(@PathVariable("phone") String phone){ return JsonResult.success(PhoneToRegionUtil.getPhoneAffiliationInfo(phone)); }
在瀏覽器中輸入地址,添加號(hào)碼:
通過(guò)測(cè)試,引用谷歌提供的包,可以解決我們的需求。
哈哈,可以不用加班.......
2、基于CSV文件
雖然引入谷歌的可以搞定需求了,但是作為程序員總要想想還有沒(méi)有其他方式實(shí)現(xiàn)?這不又找一種方式。哈哈
其實(shí)我們的手機(jī)號(hào)是有規(guī)律可循的:
1、前3位:前三位的數(shù)字,其實(shí)代表的是運(yùn)營(yíng)商。不同的運(yùn)營(yíng)商會(huì)提供不同的號(hào)段。比如:我的手機(jī)號(hào)是135開(kāi)頭就是移動(dòng)提供的。移動(dòng)除了提供135號(hào)段外,還有其他各種號(hào)段,如134,137等;聯(lián)通則提供了:130,131等號(hào)段;電信呢,提供了133,153等號(hào)段。
2、前7位:前7位則是可以確定手機(jī)號(hào)的歸屬地,例如:我的手機(jī)號(hào)前7位是1350154,則可以確定是廣東省廣州市。
既然我們知道了手機(jī)號(hào)的一些規(guī)律,那么如果有一份這樣的文檔,我們是不是就可以基于這份文檔進(jìn)行歸屬地的查詢呢?
還真有這樣的一份文檔,我在網(wǎng)上找到一份4年前的CSV文檔。如圖:
既然有這份文檔那我們就好實(shí)現(xiàn)了該功能。
2.1 讀取CSV文件
只所以寫(xiě)讀取CSV文件,是因?yàn)樽x取到這些信息后,想怎么查詢就由我們自己說(shuō)了算了。可以將數(shù)據(jù)存儲(chǔ)到數(shù)據(jù)庫(kù)查詢,也可以放在redis中查詢。下面我們基于redis的查詢來(lái)實(shí)現(xiàn)歸屬功能。
將CSV文件讀取到redis中。
/** * @author: jiangjs * @description: 服務(wù)啟動(dòng)后,加載數(shù)據(jù)到緩存 * @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,"讀取手機(jī)號(hào)信息文件為空"); 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("獲取手機(jī)號(hào)信息報(bào)錯(cuò)"); } } } }
我們?cè)谙到y(tǒng)啟動(dòng)后,自動(dòng)將CSV數(shù)據(jù)加載到redis中,當(dāng)然通過(guò)@ConditionalOnProperty可以來(lái)自行決定要不要加載到內(nèi)存中。不知道@ConditionalOnProperty注解使用的小伙伴去我的主頁(yè)可以找到這篇文章來(lái)了解。
2.2 創(chuàng)建工具
數(shù)據(jù)被加載到內(nèi)存后,那么我們就可以編寫(xiě)工具類來(lái)進(jìn)行獲取手機(jī)歸屬地。
/** * @author: jiangjs * @description: 讀取CSV文件,根據(jù)手機(jī)號(hào)前7位進(jìn)行匹配 * @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; /** * 根據(jù)手機(jī)號(hào)獲取手機(jī)歸屬地 * @param phone 手機(jī)號(hào) * @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 獲取歸屬地
有了工具類,那我們來(lái)測(cè)試一下。
直接在Controller層中編寫(xiě)接口:
@Resource private PhoneToRegionCsvUtil phoneToRegionCsvUtil; @GetMapping("/getPhoneGeoInfoByCsv.do/{phone}") public JsonResult<?> getPhoneGeoInfoByCsv(@PathVariable("phone") String phone){ return JsonResult.success(phoneToRegionCsvUtil.getPhoneToRegion(phone)); }
瀏覽器中訪問(wèn):
至此我們也可以正常的獲取到手機(jī)號(hào)的歸屬地。
3、頁(yè)面抓取
頁(yè)面抓取這種方式,跟我之前的《SpringBoot通過(guò)ip獲取歸屬地的幾種方式》中的頁(yè)面抓取方式是一樣的,在這就不跟大家詳細(xì)介紹了。
總結(jié)
文中介紹了三種方式進(jìn)行手機(jī)號(hào)查詢歸屬地的方式。
第一種:基于谷歌提供的國(guó)際解析包,引入后不用額外引入其他的東西,只需要寫(xiě)工具類即可,查詢速度也比較快。
第二種:基于CSV文件的,不用額外引入具體的包,但是要引入CSV文件,大小在12M多,當(dāng)然也可以將文件放在磁盤(pán)里,這樣不用擔(dān)心部署包過(guò)大。如果是基于內(nèi)存查詢的話,則需要依賴redis,增加了難度。
第三種:這個(gè)就不推薦了,畢竟依賴于第三方,如果服務(wù)掛了的話就沒(méi)法使用了。如果用戶量大的話,很可能會(huì)被第三方......,大家都懂的。
我在應(yīng)用就是使用了第一種方式。
以上就是SpringBoot實(shí)現(xiàn)根據(jù)手機(jī)號(hào)獲取歸屬地的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot手機(jī)號(hào)獲取歸屬地的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java Cache詳解及簡(jiǎn)單實(shí)現(xiàn)
這篇文章主要介紹了 Java Cache詳解及簡(jiǎn)單實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下2017-02-02通過(guò)第三方接口發(fā)送短信驗(yàn)證碼/短信通知(推薦)
這篇文章主要介紹了通過(guò)第三方接口發(fā)送短信驗(yàn)證碼/短信通知(推薦)的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-08-08SpringBoot返回前端Long類型字段丟失精度問(wèn)題及解決方案
Java服務(wù)端返回Long整型數(shù)據(jù)給前端,JS會(huì)自動(dòng)轉(zhuǎn)換為Number類型,本文主要介紹了SpringBoot返回前端Long類型字段丟失精度問(wèn)題及解決方案,感興趣的可以了解一下2024-03-03Java設(shè)計(jì)模式中的裝飾器模式簡(jiǎn)析
這篇文章主要介紹了Java設(shè)計(jì)模式中的裝飾器模式簡(jiǎn)析,裝飾模式能夠?qū)崿F(xiàn)動(dòng)態(tài)的為對(duì)象添加功能,是從一個(gè)對(duì)象外部來(lái)給對(duì)象添加功能,通常給對(duì)象添加功能,要么直接修改對(duì)象添加相應(yīng)的功能,要么派生對(duì)應(yīng)的子類來(lái)擴(kuò)展,抑或是使用對(duì)象組合的方式,需要的朋友可以參考下2023-12-12SpringBoot中事務(wù)失效的六個(gè)原因解析
這篇文章主要介紹了SpringBoot中事務(wù)失效的六個(gè)原因解析,由于Spring的事務(wù)是基于AOP的方式結(jié)合動(dòng)態(tài)代理來(lái)實(shí)現(xiàn)的,因此事務(wù)方法一定要是public的,這樣才能便于被Spring做事務(wù)的代理和增強(qiáng),需要的朋友可以參考下2023-10-10多線程-lock與lockInterruptibly的區(qū)別及說(shuō)明
文章主要討論了Java中ReentrantLock的lock和lockInterruptibly方法的區(qū)別,以及AQS中的雙向鏈表設(shè)計(jì),lock方法不響應(yīng)中斷,而lockInterruptibly方法會(huì)響應(yīng)中斷,AQS的雙向鏈表設(shè)計(jì)使得線程管理更加高效和靈活,適用于高并發(fā)場(chǎng)景2025-02-02