java如何實(shí)現(xiàn)獲取客戶端ip地址的示例代碼
一、前言
環(huán)境:jdk1.8 + idea2019.3 + Windows10
二、摘要
項(xiàng)目開發(fā)中,在日常處理操作日志時(shí),基本都會(huì)通過aop切面來實(shí)現(xiàn)用戶操作日志的記錄,但涉及到記錄操作日志,想必肯定會(huì)有一項(xiàng)是基本都會(huì)參考記錄的,那就是操作人的客戶端ip地址,日后查"臟"也方便。
那么問題來了。具體怎么獲取客戶端的ip地址?。抗?,這就是我這期的教學(xué)內(nèi)容,若是有的小伙伴知道怎么獲取,那么我要表揚(yáng)你,但是實(shí)現(xiàn)思路是否與我有同異?所以你也可以嘗試看下bug菌是如何實(shí)現(xiàn)的吧。
接下來,我就開始教學(xué)啦,你們可得用心聽。
三、java實(shí)現(xiàn)獲取客戶端ip
第一步:我們先來定義一個(gè)工具類,專門用于封裝ip相關(guān)的方法類。
package com.example.review.util; import org.apache.commons.lang3.StringUtils; import javax.servlet.http.HttpServletRequest; /** * ip相關(guān)工具類 * * @Author luoYong * @Date 2022-03-30 17:16 */ public class IpUtils { }
第二步:也就是此期教學(xué)的核心所在了,那究竟怎么獲取客戶端ip呢?我也就不賣關(guān)子了。
通常獲取客戶端的IP地址都是通過 request.getRemoteAddr() ,對吧,但是你有想過么,如今基本系統(tǒng)都會(huì)進(jìn)行域名代理等,比如通過 Apache,Squid 等反向代理軟件,使用getRemoteAddr() 根本無法再獲取到客戶端的真實(shí) IP 地址了。
為什么加了代理就獲取不到了呢?這是因?yàn)樵诳蛻舳撕头?wù)之間增加了中間代理,因此服務(wù)器是無法直接拿到客戶端的 IP地址,服務(wù)器端應(yīng)用也無法直接通過轉(zhuǎn)發(fā)請求的地址返回給客戶端,基本這種獲取方式就直接被pass掉了。
如果還不理解,可以看下下邊的代理流程示意圖,你就會(huì)理解啦。
如果只局限系統(tǒng)不做代理,那肯定是ok的。那系統(tǒng)基本不可能不做代理的呀,所以咋辦?別著急,我會(huì)告訴你怎么玩兒,如果使用了我的教學(xué)方法還獲取不到,請你來揍我,好吧?
如下是具體獲取客戶端ip方法類:僅供大家參考。
/*** * 獲取客戶端ip地址 * @param request */ public static String getIP(final HttpServletRequest request) throws Exception { if (request == null) { throw (new Exception("getIpAddr method HttpServletRequest Object is null")); } String ipStr = request.getHeader("x-forwarded-for"); if (StringUtils.isBlank(ipStr) || "unknown".equalsIgnoreCase(ipStr)) { ipStr = request.getHeader("Proxy-Client-IP"); } if (StringUtils.isBlank(ipStr) || "unknown".equalsIgnoreCase(ipStr)) { ipStr = request.getHeader("WL-Proxy-Client-IP"); } if (StringUtils.isBlank(ipStr) || "unknown".equalsIgnoreCase(ipStr)) { ipStr = request.getRemoteAddr(); } // 多個(gè)路由時(shí),取第一個(gè)非unknown的ip final String[] arr = ipStr.split(","); for (final String str : arr) { if (!"unknown".equalsIgnoreCase(str)) { ipStr = str; break; } } //目的是將localhost訪問對應(yīng)的ip 0:0:0:0:0:0:0:1 轉(zhuǎn)成 127.0.0.1。 return ipStr.equals("0:0:0:0:0:0:0:1") ? "127.0.0.1" : ipStr; }
需要注意的是,自己本地開發(fā),基本都是通過使用 [ localhost ]標(biāo)準(zhǔn)主機(jī)名來進(jìn)行接口訪問的,對吧,那么你就會(huì)特別好奇,數(shù)據(jù)庫記錄日志 ip 這一欄,ip有的竟然是 0:0:0:0:0:0:0:1。
為什么獲取到的ip是這樣呢?這是因?yàn)?0:0:0:0:0:0:0:1 是[ ipv6 ]的表現(xiàn)形式,對應(yīng)[ ipv4 ]來說相當(dāng)于127.0.0.1,也就是本機(jī)。所以我在最后通過三目運(yùn)算強(qiáng)行把ipv6地址轉(zhuǎn)成了ipv4地址。這樣講大家可否能理解?不理解就回頭好好惡補(bǔ)一下計(jì)算機(jī)網(wǎng)絡(luò),這也不是不可以。
如下分別是通過 localhost 及實(shí)際ip分別進(jìn)行接口訪問,數(shù)據(jù)庫保存的操作日志記錄。
附上獲取本地ip數(shù)據(jù)庫截圖:
好啦,以上就是的通過java實(shí)現(xiàn)獲取客戶端ip的全部教學(xué)啦。
相關(guān)文章
shade解決mybatis包沖突問題及項(xiàng)目引用的方法
這篇文章主要介紹了shade解決mybatis包沖突問題及項(xiàng)目引用的方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08Jenkins+Docker+Gitee+SpringBoot自動(dòng)化部署
本文主要介紹了Jenkins+Docker+Gitee+SpringBoot自動(dòng)化部署,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03java文件下載設(shè)置中文名稱的實(shí)例(response.addHeader)
下面小編就為大家分享一篇java文件下載設(shè)置中文名稱的實(shí)例(response.addHeader),具有很好的參考價(jià)值,希望對大家有所幫助2017-12-12mybatisplus?selectOne查詢,有數(shù)據(jù),但返回為null問題
這篇文章主要介紹了mybatisplus?selectOne查詢,有數(shù)據(jù),但返回為null問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11Intellij IDEA 配置Subversion插件實(shí)現(xiàn)步驟詳解
這篇文章主要介紹了Intellij IDEA 配置Subversion插件實(shí)現(xiàn)步驟詳解的相關(guān)資料,需要的朋友可以參考下2017-05-05MyBatis利用MyCat實(shí)現(xiàn)多租戶的簡單思路分享
這篇文章主要給大家介紹了關(guān)于MyBatis利用MyCat實(shí)現(xiàn)多租戶的簡單思路的相關(guān)資料,文中的多租戶是基于多數(shù)據(jù)庫進(jìn)行實(shí)現(xiàn)的,數(shù)據(jù)是通過不同數(shù)據(jù)庫進(jìn)行隔離,需要的朋友可以參考借鑒,下面來一起看看吧。2017-06-06