SpringBoot項目中獲取IP地址

提示:文章寫完后,目錄可以自動生成,如何生成可參考右邊的幫助文檔

文章目錄


前言

OkHttp 是一個由 Square 開發的高效、現代的 HTTP 客戶端庫,用于 Android 和 Java 應用程序。它支持 HTTP/2 和 SPDY 等現代網絡協議,并提供了多種功能和優化,使其成為處理網絡請求的流行選擇。這次項目中我將會使用OkHttp來發送網絡請求


一、OkHttp是什么?

OkHttp 是一個由 Square 開發的高效、現代的 HTTP 客戶端庫,用于 Android 和 Java 應用程序。

二、使用步驟

1.OkHttp請求代碼

package com.easybbs.utils;import com.easybbs.entity.enums.ResponseCodeEnum;
import com.easybbs.exception.BusinessException;
import okhttp3.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSocketFactory;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
import java.io.IOException;
import java.net.ConnectException;
import java.net.SocketTimeoutException;
import java.security.SecureRandom;
import java.security.cert.X509Certificate;
import java.util.Map;
import java.util.concurrent.TimeUnit;public class OKHttpUtils {/*** 請求超時時間5秒*/private static final int TIME_OUT_SECONDS = 5;private static Logger logger = LoggerFactory.getLogger(OKHttpUtils.class);private static OkHttpClient.Builder getClientBuilder() {OkHttpClient.Builder clientBuilder = new OkHttpClient.Builder().followRedirects(false).addInterceptor(new RedirectInterceptor()).retryOnConnectionFailure(false);clientBuilder.connectTimeout(TIME_OUT_SECONDS, TimeUnit.SECONDS).readTimeout(TIME_OUT_SECONDS, TimeUnit.SECONDS);clientBuilder.sslSocketFactory(createSSLSocketFactory()).hostnameVerifier((hostname, session) -> true);return clientBuilder;}private static Request.Builder getRequestBuilder(Map<String, String> header) {Request.Builder requestBuilder = new Request.Builder();if (null != header) {for (Map.Entry<String, String> map : header.entrySet()) {String key = map.getKey();String value;if (map.getValue() == null) {value = "";} else {value = map.getValue();}requestBuilder.addHeader(key, value);}}return requestBuilder;}private static FormBody.Builder getBuilder(Map<String, String> params) {FormBody.Builder builder = new FormBody.Builder();if (params == null) {return builder;}for (Map.Entry<String, String> map : params.entrySet()) {String key = map.getKey();String value;if (map.getValue() == null) {value = "";} else {value = map.getValue();}builder.add(key, value);}return builder;}public static String getRequest(String url) throws BusinessException {ResponseBody responseBody = null;try {OkHttpClient.Builder clientBuilder = getClientBuilder();Request.Builder requestBuilder = getRequestBuilder(null);OkHttpClient client = clientBuilder.build();Request request = requestBuilder.url(url).build();Response response = client.newCall(request).execute();responseBody = response.body();return responseBody.string();} catch (SocketTimeoutException | ConnectException e) {logger.error("OKhttp POST 請求超時,url:{}", url, e);throw new BusinessException(ResponseCodeEnum.CODE_900);} catch (Exception e) {logger.error("OKhttp GET 請求異常", e);return null;} finally {if (responseBody != null) {responseBody.close();}}}public static String postRequest(String url, Map<String, String> header, Map<String, String> params) throws BusinessException {ResponseBody responseBody = null;try {OkHttpClient.Builder clientBuilder = getClientBuilder();Request.Builder requestBuilder = getRequestBuilder(header);FormBody.Builder builder = getBuilder(params);OkHttpClient client = clientBuilder.build();RequestBody requestBody = builder.build();Request request = requestBuilder.url(url).post(requestBody).build();Response response = client.newCall(request).execute();responseBody = response.body();String responseStr = responseBody.string();return responseStr;} catch (SocketTimeoutException | ConnectException e) {logger.error("OKhttp POST 請求超時,url:{}", url, e);throw new BusinessException(ResponseCodeEnum.CODE_900);} catch (Exception e) {logger.error("OKhttp POST 請求異常,url:{}", url, e);return null;} finally {if (responseBody != null) {responseBody.close();}}}private static SSLSocketFactory createSSLSocketFactory() {SSLSocketFactory ssfFactory = null;try {SSLContext sc = SSLContext.getInstance("TLSv1.2");sc.init(null, new TrustManager[]{new TrustAllCerts()}, new SecureRandom());ssfFactory = sc.getSocketFactory();} catch (Exception e) {e.printStackTrace();}return ssfFactory;}}class TrustAllCerts implements X509TrustManager {@Overridepublic void checkClientTrusted(X509Certificate[] chain, String authType) {}@Overridepublic void checkServerTrusted(X509Certificate[] chain, String authType) {}@Overridepublic X509Certificate[] getAcceptedIssuers() {return new X509Certificate[0];}
}class RedirectInterceptor implements Interceptor {private static Logger logger = LoggerFactory.getLogger(RedirectInterceptor.class);@Overridepublic Response intercept(Chain chain) throws IOException {Request request = chain.request();Response response = chain.proceed(request);int code = response.code();if (code == 307 || code == 301 || code == 302) {//獲取重定向的地址String location = response.headers().get("Location");logger.info("重定向地址,location:{}", location);//重新構建請求Request newRequest = request.newBuilder().url(location).build();response = chain.proceed(newRequest);}return response;}
}

2.獲取Ip地址

代碼如下(示例):這個代碼只能獲取到省份地址,具體信息請看下面的詳細訪問

public String getIpAddress(String ip){try {String url = "http://whois.pconline.com.cn/ipJson.jsp?json=true&ip=" + ip;String responseJson = OKHttpUtils.getRequest(url);if(null == responseJson){return Constants.NO_ADDRESS;}Map<String,String> addressInfo = JsonUtils.convertJson2Obj(responseJson,Map.class);return addressInfo.get("pro");}catch (Exception e){logger.error("獲取ip地址失敗",e);}return Constants.NO_ADDRESS;}

?3.Controller層獲取Ip地址

@RequestMapping("/login")public String login(HttpServletRequest request){String ip = getIpAddr(request)return getIpAddress(ip);}/*** 獲取客戶端IP地址* 由于客戶端的IP地址可能通過多個代理層轉發,因此需要檢查多個HTTP頭字段以獲取真實IP。* 此方法首先檢查“x-forwarded-for”頭,這是最常用的代理頭,然后嘗試其他不那么常見的頭字段。* 如果所有嘗試都失敗,則回退到使用請求的遠程地址。** @param request HttpServletRequest對象,用于獲取客戶端IP地址。* @return 客戶端的IP地址字符串。如果無法確定客戶端IP,則返回請求的遠程地址。*/protected String getIpAddr(HttpServletRequest request) {// 嘗試獲取“x-forwarded-for”頭,這是最常用的代理頭字段。String ip = request.getHeader("x-forwarded-for");// 檢查“x-forwarded-for”頭是否有效,并提取第一個IP地址。if (ip != null && ip.length() != 0 && !"unknown".equalsIgnoreCase(ip)) {// 多次反向代理后會有多個ip值,第一個ip才是真實ipif (ip.indexOf(",") != -1) {ip = ip.split(",")[0];}}// 如果“x-forwarded-for”頭無效,嘗試其他不那么常見的代理頭字段。if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("Proxy-Client-IP");}if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("WL-Proxy-Client-IP");}if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("HTTP_CLIENT_IP");}if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("HTTP_X_FORWARDED_FOR");}if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("X-Real-IP");}// 如果所有代理頭字段都無效,回退到使用請求的遠程地址作為客戶端IP。if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {ip = request.getRemoteAddr();}// 返回獲取到的IP地址,無論它是通過代理頭還是直接從請求中獲取。return ip;}

?獲取信息如上,可以自行獲取其他信息


總結

本次項目總結如何獲取Ip地址

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/37034.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/37034.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/37034.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Jmeter 進行http接口測試

&#x1f345; 視頻學習&#xff1a;文末有免費的配套視頻可觀看 &#x1f345; 點擊文末小卡片&#xff0c;免費獲取軟件測試全套資料&#xff0c;資料在手&#xff0c;漲薪更快 本文主要針對http接口進行測試&#xff0c;使用 jmeter工具實現。 Jmeter工具設計之初是用于做性…

如何用Vue3和Plotly.js繪制動態3D圖表?

本文由ScriptEcho平臺提供技術支持 項目地址&#xff1a;傳送門 Plotly.js: 使用Vue.js動態加載數據并繪制圖表 應用場景 在數據可視化應用中&#xff0c;需要將數據動態加載到圖表中并進行實時更新。本文將展示如何使用Plotly.js和Vue.js實現這一功能&#xff0c;從加載外…

MobPush iOS端海外推送最佳實現

推送注冊 在AppDelegate里進行SDK初始化&#xff08;也可以在Info.plist文件中進行AppKey&#xff0c;AppSecret的配置&#xff09;并對通知功能進行注冊以及設置推送的環境和切換海外服務器等&#xff0c;參考如下步驟代碼&#xff1a; <span style"background-colo…

【深度學習】圖形模型基礎(1):使用潛在變量模型進行數據分析的box循環

1.緒論 探索數據背后的隱藏規律&#xff0c;這不僅是數據分析的藝術&#xff0c;更是概率模型展現其威力的舞臺。在這一過程中&#xff0c;潛在變量模型尤為關鍵&#xff0c;它成為了數據驅動問題解決的核心引擎。潛在變量模型的基本理念在于&#xff0c;那些看似復雜、雜亂無…

又是一篇關于GD32堆棧的梳理+FreeRTOS的空間

GD32F103CB&#xff1a;SRAM 20K&#xff08;0x5000&#xff09; 這篇文章主要想講清楚幾個事情&#xff1a; 1、啟動文件Stack_Size、Heap_Size的大小設置有啥影響&#xff1b; 2、FreeRTOS的內存&#xff1a;FreeRTOSConfig.h文件configTOTAL_HEAP_SIZE&#xff1b; 問題2…

訊飛星火V4.0 發布,全面對標GPT-4 Turbo

6月27日&#xff0c;訊飛星火V4.0如期而至&#xff0c;升級成為更懂你的AI助手。 七大核心能力持續突破&#xff0c;全面對標GPT-4 Turbo。在8個國際主流測試集中排名第一&#xff0c;訊飛星火以一份惹眼的成績單&#xff0c;成為國內大模型的先行者。 發布會現場&#xff0c…

一個簡單易用,跨平臺的通用版本管理器,VMR

項目主頁&#xff1a;https://vdocs.vmr.us.kg/zh-cn/ 歡迎PR&#xff0c;Issue&#xff0c;Star。 類別&#xff1a;Go 項目標題&#xff1a;一個簡單易用&#xff0c;跨平臺卻非常強大的通用版本管理器&#xff0c;VMR 項目描述&#xff1a; 目前各種SDK版本管理器存在以下…

用數組模擬棧實現遞歸函數模擬

做算法課設時候看到題目要求模擬函數遞歸時候棧的入棧出棧過程。本來想著直接調用系統遞歸函數即可&#xff0c;可是發現系統函數棧的空間非常小大約只有3000層&#xff0c;很容易爆棧。于是便有了用棧去模擬遞歸函數的想法&#xff0c;但是上網查了下貌似相關代碼比較少&#…

小馬搬運物品-第13屆藍橋杯省賽Python真題精選

[導讀]&#xff1a;超平老師的Scratch藍橋杯真題解讀系列在推出之后&#xff0c;受到了廣大老師和家長的好評&#xff0c;非常感謝各位的認可和厚愛。作為回饋&#xff0c;超平老師計劃推出《Python藍橋杯真題解析100講》&#xff0c;這是解讀系列的第89講。 小馬搬運物品&…

如何與Honda建立EDI連接?

你是本田Honda的新供應商&#xff0c;需要具備EDI電子數據交換功能嗎&#xff1f;在與本田Honda交換EDI消息時需要幫助嗎&#xff1f;本文將帶你快速了解Honda的EDI需求&#xff0c;明確EDI對接需要完成的工作。 項目背景 本田是一家世界領先的汽車制造商&#xff0c;在全球2…

倉庫選址問題【數學規劃的應用(含代碼)】阿里達院MindOpt

本文主要講述使用MindOpt工具優化倉庫選址的數學規劃問題。 視頻講解&#x1f448;&#x1f448;&#x1f448;&#x1f448;&#x1f448;&#x1f448;&#x1f448;&#x1f448;&#x1f448; 一、案例場景 倉庫選址問題在現代物流和供應鏈管理中具有重要的應用。因為倉庫…

《數據結構與算法基礎 by王卓老師》學習筆記——2.2線性表的案例引入

案例一&#xff1a;一元多項式的運算 案例二&#xff1a;稀疏多項式的運算 案例三&#xff1a;圖書信息管理系統 總結

【Leetcode】520. 檢測大寫字母

文章目錄 題目思路代碼復雜度分析時間復雜度空間復雜度 結果總結 題目 題目鏈接&#x1f517;我們定義&#xff0c;在以下情況時&#xff0c;單詞的大寫用法是正確的&#xff1a; 全部字母都是大寫&#xff0c;比如 “USA” 。單詞中所有字母都不是大寫&#xff0c;比如 “le…

Mybatis入門——語法詳解:基礎使用、增刪改查、起別名、解決問題、注釋、動態查詢,從入門到進階

文章目錄 1.基礎使用1.添加依賴2.在resouces文件下新建xml文件db.properties3.在resouces文件下新建xml文件mybatis-config-xml4.創建一個MybatisUtils工具類5.創建xml文件XxxMapper.xml映射dao層接口6.添加日志5.測試 2.增刪改查1.select2.delete3.update4.insert5.模糊查詢6.…

同心創建 共踐食安 | 趙夢澈榮獲食品安全大使

“民族要復興&#xff0c;鄉村必振興”&#xff0c;為深入貫徹落實國家鄉村振興戰略&#xff0c;推進鄉村全面振興不斷取得新成效&#xff0c;助力全國優質食品農產品的宣傳推廣、市場營銷、品牌創建工作&#xff0c;由中國食品安全報社主辦&#xff0c;商業發展中心、健康中國…

python數據分析與可視化一

公共部分 # 引入數據分析工具 Pandas import pandas as pd # 引入數據可視化工具 Matplotlib import matplotlib.pyplot as plt # 引入數據可視化工具 Seaborn (基于matplotlib) import seaborn as sns # 解決輸出時的列名對齊問題 pd.set_option(display.unicode.east_…

Python多線程編程詳解

Python多線程編程詳解 大家好&#xff0c;我是免費搭建查券返利機器人省錢賺傭金就用微賺淘客系統3.0的小編&#xff0c;也是冬天不穿秋褲&#xff0c;天冷也要風度的程序猿&#xff01; 多線程編程是利用計算機多核心和多線程處理器的優勢&#xff0c;提高程序并發性能的重要…

如何申請免費SSL證書以消除訪問網站顯示連接不安全提醒

在當今互聯網時代&#xff0c;網絡安全已成為一個不可忽視的問題。當用戶瀏覽一些網站時&#xff0c;有時會看到瀏覽器地址欄出現“不安全”的提示&#xff0c;這意味著該網站沒有安裝SSL證書&#xff0c;數據傳輸可能存在風險。那么&#xff0c;如何消除這種不安全提醒&#x…

2024年6月,Altair被Gartner魔力象限評為數據科學與機器學習平臺領導者

Altair 因其愿景完整性和執行能力被評為領導者 2024 年 6 月 20 日&#xff0c;Altair&#xff08;納斯達克股票代碼&#xff1a;ALTR&#xff09;宣布&#xff0c;Altair RapidMiner 被 Gartner Magic Quadrant?&#xff08;魔力象限&#xff09;評為數據科學與機器學習平臺領…

SpringBoot配置參數獲取

1、使用Value注解 import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component;Component public class MyBean {Value("${myapp.name}") private String appName;public void printAppName() {System.out.print…