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

介紹

細心的小夥伴可能會發現,抖音新上線瞭IP屬地的功能,小夥伴在發表動態、發表評論以及聊天的時候,都會顯示自己的IP屬地信息

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

  • 通過 HttpServletRequest 對象,獲取用戶的IP地址
  • 通過 IP 地址,獲取對應的省份、城市

首先需要寫一個IP獲取的工具類,因為每一次用戶的Request請求,都會攜帶上請求的IP地址放到請求頭中。

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")) {
                // 根據網卡取本機配置的IP
                try {
                    InetAddress inet = InetAddress.getLocalHost();
                    ipAddress = inet.getHostAddress();
                } catch (UnknownHostException e) {
                    log.error("根據網卡獲取本機配置的IP異常", e);
                }
            }
        }
        // 對於通過多個代理的情況,第一個IP為客戶端真實IP,多個IP按照','分割
        if (ipAddress != null && ipAddress.indexOf(",") > 0) {
            ipAddress = ipAddress.split(",")[0];
        }
        return ipAddress;
    }
}

這裡有三個名詞,分別是

  • X-Forwarded-For:一個 HTTP擴展頭部,主要是為瞭讓Web服務器獲取訪問用戶的真實IP地址。每個IP地址,每個值通過逗號+空格分開,最左邊是最原始客戶端的IP地址,中間如果有多層代理,每⼀層代理會將連接它的客戶端IP追加在X-Forwarded-For右邊。
  • X-Real-IP:一般隻記錄真實發出請求的客戶端IP
  • Proxy-Client-IP:這個一般是經過Apache http服務器的請求才會有,用Apache http做代理時一般會加上Proxy-Client-IP請求頭
  • WL-Proxy-Client-IP:也是通過 Apache http 服務器,在weblogic插件加上的頭。

在我們獲取到用戶的IP地址後,那麼就可以獲取對應的ip信息瞭

我在Github沖浪的時候,發現瞭Ip2region項目。

一個準確率99.9%的離線IP地址定位庫,0.0x毫秒級查詢,ip2region.db數據庫隻有數MB,提供瞭 java,php,c,python,nodejs,golang,c# 等查詢綁定和Binary,B樹,內存三種查詢算法。

數據聚合瞭一些知名ip到地名查詢提供商的數據,這些是他們官方的的準確率,經測試著實比經典的純真IP定位準確一些。ip2region的數據聚合自以下服務商的開放API或者數據。

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

備註:如果上述開放API或者數據都不給開放數據時ip2region將停止數據的更新服務。

每條ip數據段都固定瞭格式:

_城市Id|國傢|區域|省份|城市|ISP_

隻有中國的數據精確到瞭城市,其他國傢有部分數據隻能定位到國傢,後前的選項全部是0,已經包含瞭全部你能查到的大大小小的國傢

生成的數據庫文件ip2region.db隻有幾MB,最小的版本隻有1.5MB,隨著數據的詳細度增加數據庫的大小也慢慢增大,目前還沒超過8MB。

內置的三種查詢算法

全部的查詢客戶端單次查詢都在0.x毫秒級別,內置瞭三種查詢算法

  • memory算法:整個數據庫全部載入內存,單次查詢都在0.1x毫秒內,C語言的客戶端單次查詢在0.00x毫秒級別。
  • binary算法:基於二分查找,基於ip2region.db文件,不需要載入內存,單次查詢在0.x毫秒級別。
  • b-tree算法:基於btree算法,基於ip2region.db文件,不需要載入內存,單詞查詢在0.x毫秒級別,比binary算法更快。

ip2region安裝

下面,就讓我們給項目引入ip2region,進行ip信息轉換吧

首先引入maven依賴

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

然後編寫一個工具類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();
    }
}

在加載的時候,需要下載倉庫中的ip2region.db文件,然後放到resource目錄下

然後,通過內置的三種算法,分別轉換用戶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,加載內存(最快)
        //DbSearcher.MEMORY_ALGORITYM
        try {
            // 使用靜態代碼塊,減少文件讀取操作
//            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函數進行測試,發現可以正常的解析出ip信息

由於 ip 屬地在國內的話,隻會展示省份,而國外的話,隻會展示國傢。所以我們還需要對這個方法進行一下封裝,得到獲取 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) {
            // 國內的顯示到具體的省
            if ("中國".equals(cityList[0])) {
                if (cityList.length > 1) {
                    return cityList[1];
                }
            }
            // 國外顯示到國傢
            return cityList[0];
        }
    }
    return "未知";
}

下面,我們在找一個 國外的IP測試一下效果。可以看到已經能夠正常的顯示IP屬地信息瞭~

到這裡如果獲取用戶的 IP 屬地已經完成啦,如果想要瞭解關於更多ip2region的功能,歡迎訪問其Github地址進行學習。

項目地址

https://github.com/lionsoul2014/ip2region

到此這篇關於Java獲取用戶IP屬地模擬抖音詳解的文章就介紹到這瞭,更多相關Java IP屬地內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: