springboot獲取真實(shí)ip地址的方法實(shí)例
需求是用戶(hù)在登錄時(shí),密碼輸錯(cuò)了5次以上要將這個(gè)ip禁用,所以如題,我需要在登錄接口代碼中獲得這次請(qǐng)求的原始ip地址。禁用ip的邏輯已經(jīng)寫(xiě)好了,目前痛點(diǎn)是獲取ip地址,也正是這塊代碼讓我被領(lǐng)導(dǎo)叼了。
既然項(xiàng)目一般都是走nginx和網(wǎng)關(guān)代理,那肯定得百度了,于是百度的代碼如下:
/**
* 獲取ip地址
*
* @param request
* @return
*/
public String getIpAddr(HttpServletRequest request) {
String ip = request.getHeader("X-Real-IP");
if (ip != null && !"".equals(ip) && !"unknown".equalsIgnoreCase(ip)) {
return ip;
}
ip = request.getHeader("X-Forwarded-For");
if (ip != null && !"".equals(ip) && !"unknown".equalsIgnoreCase(ip)) {
int index = ip.indexOf(',');
if (index != -1) {
return ip.substring(0, index);
} else {
return ip;
}
} else {
return request.getRemoteAddr();
}
}
然后我模擬2個(gè)用戶(hù)測(cè)了一下(測(cè)試結(jié)果為化名):
- 公司電腦本地啟服務(wù),注冊(cè)到公司開(kāi)發(fā)環(huán)境的nacos上,通過(guò)網(wǎng)關(guān)調(diào)用此登錄接口
- 用另一臺(tái)電腦連公司vpn,其余步驟如上。
- request.getHeader("X-Real-IP") 獲取的ip地址始終固定是:115.1.2.3
- request.getHeader("X-Forwarded-For")獲取的結(jié)果固定是: 115.1.2.3,10.x.x.x
- 結(jié)果是一個(gè)字符串,第一個(gè)值與X-Real-IP值一樣。
- 第二個(gè)值一直在動(dòng)態(tài)變化
- request.getRemoteAddr()獲取的結(jié)果固定是: 192.168.1.10
所以我很疑惑,首先看了下公司日志過(guò)濾器那塊代碼也正是用的上面的方法,完全一樣。那么說(shuō)明是一樣的。
這方法正確嗎?
大概看了下這方法邏輯就是先獲取請(qǐng)求體的realIp,獲取不到才去獲取forward啥的ip。
那我這邊為什么始終能獲取到realIp且始終固定1個(gè)值,我已經(jīng)模擬了2個(gè)用戶(hù)來(lái)登錄了。
那不是起不到不同ip的效果了嗎。
所以我懷疑是不是這個(gè)115.1.2.3是網(wǎng)關(guān)還是哪臺(tái)服務(wù)器的ip。
為什么X-Forwarded-For的第一個(gè)值與realIp一樣且第二個(gè)值一直變。且第二個(gè)值代表什么ip,為什么一直在變。此刻我懷疑是不是就是我2臺(tái)電腦的無(wú)線網(wǎng)動(dòng)態(tài)變化的ip,那么這么看我得取這第二個(gè)值了。
帶著這些疑惑,真的很菜的我繼續(xù)百度+咨詢(xún)公司運(yùn)維+最終詢(xún)問(wèn)領(lǐng)導(dǎo)。
得到了以下結(jié)論:
首先明確request.getRemoteAddr()這塊不能要。一看就是內(nèi)網(wǎng)ip,不然直接鎖死了。
在jax的告誡"去掉這塊代碼,讓其為空則不禁用,否則內(nèi)網(wǎng)鎖定ip,且很可能是用戶(hù)微服務(wù)集群2臺(tái)的ip,那就都不能登錄了"下,我刪掉了這塊代碼。
且最終咨詢(xún)了運(yùn)維之后發(fā)現(xiàn)這個(gè)固定的192.168.1.10正是開(kāi)發(fā)環(huán)境的K8S地址!那確實(shí)不能返回!
realIp固定的115.1.2.3問(wèn)了運(yùn)維大哥后知道了是真實(shí)ip地址!不是內(nèi)網(wǎng)地址!且是公司的網(wǎng)關(guān)地址。
那我明白了,我連公司vpn及我公司電腦其實(shí)都是連的公司wifi,所以最終請(qǐng)求發(fā)出去都是走的公司網(wǎng)關(guān),此網(wǎng)關(guān)不是web項(xiàng)目的網(wǎng)關(guān)!
所以最終地址都一致。那這么看滿(mǎn)足需求,1個(gè)人連的1個(gè)wifi,不能連續(xù)輸錯(cuò)密碼5次否則ip禁用,用其他方式連wifi也一樣。
“X-Forwarded-For”第一個(gè)值與其realIo一樣應(yīng)該是我只有1個(gè)代理。這邊也確實(shí)只有Nginx代理了。
所以“X-Forwarded-For”第二個(gè)值開(kāi)始則是代理服務(wù)器的地址。
咨詢(xún)了jax和讓運(yùn)維看了后發(fā)現(xiàn)是k8s的一些容器的地址。
只是這個(gè)地址為什么一直變jax也很奇怪。我測(cè)了下是每次都在變。
所以原來(lái)方法是對(duì)的,只去獲取forward的第一個(gè)值!
“X-Forwarded-For”的字面意思是“為誰(shuí)而轉(zhuǎn)發(fā)”,形式上和“Via”差不多,也是每經(jīng)過(guò)一個(gè)代理節(jié)點(diǎn)就會(huì)在字段里追加一個(gè)信息。但“Via”追加的是代理主機(jī)名(或者域名),而“X-Forwarded-For”追加的是請(qǐng)求方的 IP 地址。所以,在字段里最左邊的 IP 地址就是客戶(hù)端的地址。
“X-Real-IP”是另一種獲取客戶(hù)端真實(shí) IP 的手段,它的作用很簡(jiǎn)單,就是記錄客戶(hù)端 IP 地址,沒(méi)有中間的代理信息,相當(dāng)于是“X-Forwarded-For”的簡(jiǎn)化版。如果客戶(hù)端和源服務(wù)器之間只有一個(gè)代理,那么這兩個(gè)字段的值就是相同的。
所以最終改造后的代碼為:
/**
* 獲取真實(shí)ip地址,不返回內(nèi)網(wǎng)地址
*
* @param request
* @return
*/
public String getIpAddr(HttpServletRequest request) {
//目前則是網(wǎng)關(guān)ip
String ip = request.getHeader("X-Real-IP");
if (ip != null && !"".equals(ip) && !"unknown".equalsIgnoreCase(ip)) {
return ip;
}
ip = request.getHeader("X-Forwarded-For");
if (ip != null && !"".equals(ip) && !"unknown".equalsIgnoreCase(ip)) {
int index = ip.indexOf(',');
if (index != -1) {
//只獲取第一個(gè)值
return ip.substring(0, index);
} else {
return ip;
}
} else {
//取不到真實(shí)ip則返回空,不能返回內(nèi)網(wǎng)地址。
return "";
}
}
總結(jié)
到此這篇關(guān)于springboot獲取真實(shí)ip地址的文章就介紹到這了,更多相關(guān)springboot獲取真實(shí)ip內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- springboot實(shí)現(xiàn)獲取客戶(hù)端IP地址的示例代碼
- SpringBoot項(xiàng)目中使用OkHttp獲取IP地址的示例代碼
- SpringBoot項(xiàng)目中獲取IP地址的實(shí)現(xiàn)示例
- springboot如何獲取請(qǐng)求者的ip地址
- SpringBoot如何獲取客戶(hù)端的IP地址
- SpringBoot實(shí)現(xiàn)IP地址解析的示例代碼
- SpringBoot獲取客戶(hù)端的IP地址的實(shí)現(xiàn)示例
- SpringBoot整合Ip2region獲取IP地址和定位的詳細(xì)過(guò)程
- springboot 獲取訪問(wèn)接口的請(qǐng)求的IP地址的實(shí)現(xiàn)
- springboot獲取訪問(wèn)的ip地址的實(shí)現(xiàn)步驟
相關(guān)文章
Spring大白話(huà)之三級(jí)緩存如何解決循環(huán)依賴(lài)問(wèn)題
Spring通過(guò)三級(jí)緩存(singletonObjects、earlySingletonObjects、singletonFactories)解決單例循環(huán)依賴(lài),三級(jí)緩存使用Lambda表達(dá)式提前暴露bean的早期引用,確保在遞歸調(diào)用時(shí)能夠正確獲取對(duì)象實(shí)例,避免死循環(huán)2025-02-02
tomcat啟動(dòng)完成執(zhí)行 某個(gè)方法 定時(shí)任務(wù)(Spring)操作
這篇文章主要介紹了tomcat啟動(dòng)完成執(zhí)行 某個(gè)方法 定時(shí)任務(wù)(Spring)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
Java中Buffer緩沖區(qū)的ByteBuffer類(lèi)詳解
這篇文章主要介紹了Java中Buffer緩沖區(qū)的ByteBuffer類(lèi)詳解,ByteBuffer類(lèi)是Java NIO庫(kù)中的一個(gè)重要類(lèi),用于處理字節(jié)數(shù)據(jù),它提供了一種靈活的方式來(lái)讀取、寫(xiě)入和操作字節(jié)數(shù)據(jù),ByteBuffer類(lèi)是一個(gè)抽象類(lèi),可以通過(guò)靜態(tài)方法創(chuàng)建不同類(lèi)型的ByteBuffer對(duì)象,需要的朋友可以參考下2023-10-10
springboot Jpa多數(shù)據(jù)源(不同庫(kù))配置過(guò)程
這篇文章主要介紹了springboot Jpa多數(shù)據(jù)源(不同庫(kù))配置過(guò)程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05
通過(guò)Java實(shí)現(xiàn)添加或刪除PDF中的附件
當(dāng)我們?cè)谥谱鱌DF文件或者PPT演示文稿的時(shí)候,為了讓自己的文件更全面詳細(xì),就會(huì)在文件中添加附件。本文為大家整理了Java實(shí)現(xiàn)添加或刪除PDF中的附件的方法,需要的可以參考下2023-01-01
SpringBoot后端解決跨域問(wèn)題的3種方案分享
這篇文章主要給大家分享介紹了關(guān)于SpringBoot后端解決跨域問(wèn)題的3種方案,跨域指的是瀏覽器不能執(zhí)行其他網(wǎng)站的腳本,它是由瀏覽器的同源策略造成的,是瀏覽器施加的安全限制,需要的朋友可以參考下2023-07-07
java中判斷String類(lèi)型為空和null的幾種方法
判斷一個(gè)字符串是否為空或者為null是一個(gè)常見(jiàn)的操作,本文主要介紹了java中判斷String類(lèi)型為空和null的幾種方法,具有一定的參考價(jià)值,感興趣的可以了解一下2024-06-06

