亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java獲取用戶IP屬地模擬抖音詳解

 更新時(shí)間:2022年07月11日 09:36:42   作者:葉秋學(xué)長(zhǎng)  
細(xì)心的小伙伴可能會(huì)發(fā)現(xiàn),抖音新上線了 IP 屬地的功能,小伙伴在發(fā)表動(dòng)態(tài)、發(fā)表評(píng)論以及聊天的時(shí)候,都會(huì)顯示自己的 IP 屬地信息,本篇文章我們來(lái)模擬實(shí)現(xiàn)這一功能

介紹

細(xì)心的小伙伴可能會(huì)發(fā)現(xiàn),抖音新上線了IP屬地的功能,小伙伴在發(fā)表動(dòng)態(tài)、發(fā)表評(píng)論以及聊天的時(shí)候,都會(huì)顯示自己的IP屬地信息

下面,我就來(lái)講講,Java中是如何獲取IP屬地的,主要分為以下幾步

  • 通過 HttpServletRequest 對(duì)象,獲取用戶的IP地址
  • 通過 IP 地址,獲取對(duì)應(yīng)的省份、城市

首先需要寫一個(gè)IP獲取的工具類,因?yàn)槊恳淮斡脩舻腞equest請(qǐng)求,都會(huì)攜帶上請(qǐng)求的IP地址放到請(qǐng)求頭中。

public class IpUtil {
    public static String getIpAddr(ServerHttpRequest request) {
        HttpHeaders headers = request.getHeaders();
        String ipAddress = headers.getFirst("X-Forwarded-For");
        if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
            ipAddress = headers.getFirst("Proxy-Client-IP");
        }
        if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
            ipAddress = headers.getFirst("WL-Proxy-Client-IP");
        }
        if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
            ipAddress = request.getRemoteAddress().getAddress().getHostAddress();
            if (ipAddress.equals("127.0.0.1") || ipAddress.equals("0:0:0:0:0:0:0:1")) {
                // 根據(jù)網(wǎng)卡取本機(jī)配置的IP
                try {
                    InetAddress inet = InetAddress.getLocalHost();
                    ipAddress = inet.getHostAddress();
                } catch (UnknownHostException e) {
                    log.error("根據(jù)網(wǎng)卡獲取本機(jī)配置的IP異常", e);
                }
            }
        }
        // 對(duì)于通過多個(gè)代理的情況,第一個(gè)IP為客戶端真實(shí)IP,多個(gè)IP按照','分割
        if (ipAddress != null && ipAddress.indexOf(",") > 0) {
            ipAddress = ipAddress.split(",")[0];
        }
        return ipAddress;
    }
}

這里有三個(gè)名詞,分別是

  • X-Forwarded-For:一個(gè) HTTP擴(kuò)展頭部,主要是為了讓W(xué)eb服務(wù)器獲取訪問用戶的真實(shí)IP地址。每個(gè)IP地址,每個(gè)值通過逗號(hào)+空格分開,最左邊是最原始客戶端的IP地址,中間如果有多層代理,每?層代理會(huì)將連接它的客戶端IP追加在X-Forwarded-For右邊。
  • X-Real-IP:一般只記錄真實(shí)發(fā)出請(qǐng)求的客戶端IP
  • Proxy-Client-IP:這個(gè)一般是經(jīng)過Apache http服務(wù)器的請(qǐng)求才會(huì)有,用Apache http做代理時(shí)一般會(huì)加上Proxy-Client-IP請(qǐng)求頭
  • WL-Proxy-Client-IP:也是通過 Apache http 服務(wù)器,在weblogic插件加上的頭。

在我們獲取到用戶的IP地址后,那么就可以獲取對(duì)應(yīng)的ip信息了

我在Github沖浪的時(shí)候,發(fā)現(xiàn)了Ip2region項(xiàng)目。

一個(gè)準(zhǔn)確率99.9%的離線IP地址定位庫(kù),0.0x毫秒級(jí)查詢,ip2region.db數(shù)據(jù)庫(kù)只有數(shù)MB,提供了 java,php,c,python,nodejs,golang,c# 等查詢綁定和Binary,B樹,內(nèi)存三種查詢算法。

數(shù)據(jù)聚合了一些知名ip到地名查詢提供商的數(shù)據(jù),這些是他們官方的的準(zhǔn)確率,經(jīng)測(cè)試著實(shí)比經(jīng)典的純真IP定位準(zhǔn)確一些。ip2region的數(shù)據(jù)聚合自以下服務(wù)商的開放API或者數(shù)據(jù)。

  • 80%, 淘寶IP地址庫(kù), http://ip.taobao.com/
  • ≈10%, GeoIP, https://geoip.com/
  • ≈2%, 純真IP庫(kù), http://www.cz88.net/

備注:如果上述開放API或者數(shù)據(jù)都不給開放數(shù)據(jù)時(shí)ip2region將停止數(shù)據(jù)的更新服務(wù)。

每條ip數(shù)據(jù)段都固定了格式:

_城市Id|國(guó)家|區(qū)域|省份|城市|ISP_

只有中國(guó)的數(shù)據(jù)精確到了城市,其他國(guó)家有部分?jǐn)?shù)據(jù)只能定位到國(guó)家,后前的選項(xiàng)全部是0,已經(jīng)包含了全部你能查到的大大小小的國(guó)家

生成的數(shù)據(jù)庫(kù)文件ip2region.db只有幾MB,最小的版本只有1.5MB,隨著數(shù)據(jù)的詳細(xì)度增加數(shù)據(jù)庫(kù)的大小也慢慢增大,目前還沒超過8MB。

內(nèi)置的三種查詢算法

全部的查詢客戶端單次查詢都在0.x毫秒級(jí)別,內(nèi)置了三種查詢算法

  • memory算法:整個(gè)數(shù)據(jù)庫(kù)全部載入內(nèi)存,單次查詢都在0.1x毫秒內(nèi),C語(yǔ)言的客戶端單次查詢?cè)?.00x毫秒級(jí)別。
  • binary算法:基于二分查找,基于ip2region.db文件,不需要載入內(nèi)存,單次查詢?cè)?.x毫秒級(jí)別。
  • b-tree算法:基于btree算法,基于ip2region.db文件,不需要載入內(nèi)存,單詞查詢?cè)?.x毫秒級(jí)別,比binary算法更快。

ip2region安裝

下面,就讓我們給項(xiàng)目引入ip2region,進(jìn)行ip信息轉(zhuǎn)換吧

首先引入maven依賴

<dependency>
    <groupId>org.lionsoul</groupId>
    <artifactId>ip2region</artifactId>
    <version>1.7.2</version>
</dependency>

然后編寫一個(gè)工具類IpUtils,首先需要加載ip2region.db文件

static {
    dbPath = createFtlFileByFtlArray() + "ip2region.db";
    try {
        config = new DbConfig();
    } catch (DbMakerConfigException e) {
        e.printStackTrace();
    }
    try {
        searcher = new DbSearcher(config, dbPath);
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    }
}

在加載的時(shí)候,需要下載倉(cāng)庫(kù)中的ip2region.db文件,然后放到resource目錄下

然后,通過內(nèi)置的三種算法,分別轉(zhuǎn)換用戶ip地址

    public static String getCityInfo(String ip) {
        if (StringUtils.isEmpty(dbPath)) {
            log.error("Error: Invalid ip2region.db file");
            return null;
        }
        if(config == null || searcher == null){
            log.error("Error: DbSearcher or DbConfig is null");
            return null;
        }
        //查詢算法
        //B-tree, B樹搜索(更快)
        int algorithm = DbSearcher.BTREE_ALGORITHM;
        //Binary,使用二分搜索
        //DbSearcher.BINARY_ALGORITHM
        //Memory,加載內(nèi)存(最快)
        //DbSearcher.MEMORY_ALGORITYM
        try {
            // 使用靜態(tài)代碼塊,減少文件讀取操作
//            DbConfig config = new DbConfig();
//            DbSearcher searcher = new DbSearcher(config, dbPath);
            //define the method
            Method method = null;
            switch (algorithm) {
                case DbSearcher.BTREE_ALGORITHM:
                    method = searcher.getClass().getMethod("btreeSearch", String.class);
                    break;
                case DbSearcher.BINARY_ALGORITHM:
                    method = searcher.getClass().getMethod("binarySearch", String.class);
                    break;
                case DbSearcher.MEMORY_ALGORITYM:
                    method = searcher.getClass().getMethod("memorySearch", String.class);
                    break;
                default:
            }
            DataBlock dataBlock = null;
            if (Util.isIpAddress(ip) == false) {
                System.out.println("Error: Invalid ip address");
            }
            dataBlock = (DataBlock) method.invoke(searcher, ip);
            String ipInfo = dataBlock.getRegion();
            if (!StringUtils.isEmpty(ipInfo)) {
                ipInfo = ipInfo.replace("|0", "");
                ipInfo = ipInfo.replace("0|", "");
            }
            return ipInfo;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

下面,我們編寫main函數(shù)進(jìn)行測(cè)試,發(fā)現(xiàn)可以正常的解析出ip信息

由于 ip 屬地在國(guó)內(nèi)的話,只會(huì)展示省份,而國(guó)外的話,只會(huì)展示國(guó)家。所以我們還需要對(duì)這個(gè)方法進(jìn)行一下封裝,得到獲取 IP 屬地的信息。

/**
 * 獲取IP屬地
 * @param ip
 * @return
 */
public static String getIpPossession(String ip) {
    String cityInfo = getCityInfo(ip);
    if (!StringUtils.isEmpty(cityInfo)) {
        cityInfo = cityInfo.replace("|", " ");
        String[] cityList = cityInfo.split(" ");
        if (cityList.length > 0) {
            // 國(guó)內(nèi)的顯示到具體的省
            if ("中國(guó)".equals(cityList[0])) {
                if (cityList.length > 1) {
                    return cityList[1];
                }
            }
            // 國(guó)外顯示到國(guó)家
            return cityList[0];
        }
    }
    return "未知";
}

下面,我們?cè)谡乙粋€(gè) 國(guó)外的IP測(cè)試一下效果??梢钥吹揭呀?jīng)能夠正常的顯示IP屬地信息了~

到這里如果獲取用戶的 IP 屬地已經(jīng)完成啦,如果想要了解關(guān)于更多ip2region的功能,歡迎訪問其Github地址進(jìn)行學(xué)習(xí)。

項(xiàng)目地址

https://github.com/lionsoul2014/ip2region

到此這篇關(guān)于Java獲取用戶IP屬地模擬抖音詳解的文章就介紹到這了,更多相關(guān)Java IP屬地內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java  Iterator接口和LIstIterator接口分析

    java Iterator接口和LIstIterator接口分析

    這篇文章主要介紹了java Iterator接口和LIstIterator接口分析的相關(guān)資料,需要的朋友可以參考下
    2017-05-05
  • Java項(xiàng)目實(shí)現(xiàn)模擬ATM機(jī)

    Java項(xiàng)目實(shí)現(xiàn)模擬ATM機(jī)

    這篇文章主要為大家詳細(xì)介紹了Java項(xiàng)目實(shí)現(xiàn)模擬ATM機(jī),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-05-05
  • Java 獲取Web項(xiàng)目相對(duì)webapp地址的實(shí)例

    Java 獲取Web項(xiàng)目相對(duì)webapp地址的實(shí)例

    下面小編就為大家?guī)?lái)一篇Java 獲取Web項(xiàng)目相對(duì)webapp地址的實(shí)例。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧
    2017-11-11
  • Java中token的存儲(chǔ)和獲取實(shí)例代碼

    Java中token的存儲(chǔ)和獲取實(shí)例代碼

    關(guān)于java獲取微信Token驗(yàn)證的問題相信很多人都遇見過,尤其是對(duì)剛接觸微信開發(fā)的人來(lái)說確實(shí)有點(diǎn)棘手,下面這篇文章主要給大家介紹了關(guān)于Java中token存儲(chǔ)和獲取的相關(guān)資料,需要的朋友可以參考下
    2022-08-08
  • Java線程池submit阻塞獲取結(jié)果的實(shí)現(xiàn)原理詳解

    Java線程池submit阻塞獲取結(jié)果的實(shí)現(xiàn)原理詳解

    Java線程池中提交任務(wù)運(yùn)行,通常使用execute()方法就足夠了。那如果想要實(shí)現(xiàn)在主線程中阻塞獲取線程池任務(wù)運(yùn)行的結(jié)果,該怎么辦呢?本文就來(lái)和大家一起討論討論
    2022-10-10
  • Spring數(shù)據(jù)庫(kù)事務(wù)的實(shí)現(xiàn)機(jī)制講解

    Spring數(shù)據(jù)庫(kù)事務(wù)的實(shí)現(xiàn)機(jī)制講解

    這篇文章主要介紹了Spring數(shù)據(jù)庫(kù)事務(wù)的實(shí)現(xiàn)機(jī)制講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • SpringBoot中整合MyBatis-Plus-Join使用聯(lián)表查詢的實(shí)現(xiàn)

    SpringBoot中整合MyBatis-Plus-Join使用聯(lián)表查詢的實(shí)現(xiàn)

    本文主要介紹了SpringBoot中整合MyBatis-Plus-Join使用聯(lián)表查詢的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Java中的八種基本數(shù)據(jù)類型詳解

    Java中的八種基本數(shù)據(jù)類型詳解

    本文詳細(xì)講解了Java中的八種基本數(shù)據(jù)類型,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-03-03
  • Java中5種異步實(shí)現(xiàn)的方式詳解

    Java中5種異步實(shí)現(xiàn)的方式詳解

    同步操作如果遇到一個(gè)耗時(shí)的方法,需要阻塞等待,那么我們有沒有辦法解決呢?讓它異步執(zhí)行,下面我會(huì)詳解異步及實(shí)現(xiàn),需要的可以參考一下
    2022-09-09
  • Java遍歷Json的簡(jiǎn)單實(shí)例

    Java遍歷Json的簡(jiǎn)單實(shí)例

    這篇文章主要介紹了Java遍歷Json的簡(jiǎn)單實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-07-07

最新評(píng)論