解決HttpServletRequest無法獲取@RequestBody修飾的參數(shù)問題及分析
HttpServletRequest無法獲取@RequestBody修飾的參數(shù)
在使用springboot設(shè)計(jì)controller時(shí)我們通常會(huì)在某個(gè)請(qǐng)求如post中使用@RequestBody來修飾參數(shù)如:

在一些特殊場(chǎng)景下,我們需要在service層的代碼去拿到當(dāng)前上下文請(qǐng)求(HttpServletRequest)中的一些信息如請(qǐng)求體,這個(gè)時(shí)候被@RequestBody所修飾的請(qǐng)求是無法獲取的。
原因如下:
1、請(qǐng)求體流只能讀取一次:Servlet 規(guī)范中,HttpServletRequest 的輸入流 (getInputStream()) 是單向的,一旦被 @RequestBody 讀取后,流就會(huì)關(guān)閉,無法再次讀取。
2、@RequestBody 優(yōu)先處理:Spring MVC 在處理控制器方法時(shí),會(huì)先解析 @RequestBody,導(dǎo)致后續(xù)通過 HttpServletRequest 獲取請(qǐng)求體時(shí)為空。
下面簡(jiǎn)單演示下解決方案:
一、先編寫一個(gè)http工具類用來讀取ServletRequest的內(nèi)容
import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import javax.servlet.ServletInputStream;
import javax.servlet.ServletRequest;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.nio.charset.StandardCharsets;
import java.util.List;
/**
* http工具類
*/
@Slf4j
public class HttpUtils {
/**
* 從request獲取body的數(shù)據(jù)
*
* @param request 請(qǐng)求
* @return body數(shù)據(jù)字符串
*/
public static String getBodyStr(ServletRequest request) {
StringBuilder sb = new StringBuilder();
try {
try (ServletInputStream inputStream = request.getInputStream()) {
try (BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, StandardCharsets.UTF_8))) {
String line;
while ((line = reader.readLine()) != null) {
sb.append(line);
}
}
}
} catch (Exception e) {
log.error("get request body error: ", e);
}
return sb.toString();
}
/**
* 從request獲取body的數(shù)據(jù),并轉(zhuǎn)換成對(duì)象(適用于使用@RequestBody修飾的參數(shù))
*
* @param request request 請(qǐng)求
* @param clazz 對(duì)象類型
* @return 對(duì)象
*/
public static <T> T getBodyToObject(ServletRequest request, Class<T> clazz) {
String bodyStr = getBodyStr(request);
if (StringUtils.isBlank(bodyStr)) {
return null;
}
return JSON.parseObject(bodyStr, clazz);
}
/**
* 從request獲取body的數(shù)據(jù),并轉(zhuǎn)換成集合對(duì)象(適用于使用@RequestBody修飾的參數(shù))
*
* @param request request 請(qǐng)求
* @param clazz 集合對(duì)象類型
* @return 集合對(duì)象
*/
public static <T> List<T> getBodyToList(ServletRequest request, Class<T> clazz) {
String bodyStr = getBodyStr(request);
if (StringUtils.isBlank(bodyStr)) {
return null;
}
return JSON.parseArray(bodyStr, clazz);
}
}二、添加請(qǐng)求體復(fù)制包裝器
import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.nio.charset.StandardCharsets;
/**
* 請(qǐng)求體復(fù)制包裝器
*/
public class BodyCopyWrapper extends HttpServletRequestWrapper {
private byte[] requestBody;
public BodyCopyWrapper(HttpServletRequest request) throws IOException {
super(request);
requestBody = HttpUtils.getBodyStr(request).getBytes(StandardCharsets.UTF_8);
}
@Override
public BufferedReader getReader() throws IOException {
return new BufferedReader(new InputStreamReader(getInputStream()));
}
@Override
public ServletInputStream getInputStream() throws IOException {
final ByteArrayInputStream basis = new ByteArrayInputStream(requestBody);
return new ServletInputStream() {
@Override
public int read() throws IOException {
return basis.read();
}
@Override
public boolean isFinished() {
return false;
}
@Override
public boolean isReady() {
return false;
}
@Override
public void setReadListener(ReadListener readListener) {
}
};
}
public void setInputStream(byte[] body) {
this.requestBody = body;
}
}三、添加請(qǐng)求攔截器配置類
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.Ordered;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
/**
* 請(qǐng)求攔截器配置類
*/
@Slf4j
@Configuration
public class FilterConfig {
@Bean
public FilterRegistrationBean<RequestWrapperFilter> requestWrapperFilter() {
FilterRegistrationBean<RequestWrapperFilter> bean = new FilterRegistrationBean<>();
bean.setFilter(new RequestWrapperFilter());
bean.addUrlPatterns("/*");
bean.setOrder(Ordered.HIGHEST_PRECEDENCE);
return bean;
}
public static class RequestWrapperFilter implements Filter {
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws ServletException
, IOException {
ServletRequest requestWrapper = null;
if (request instanceof HttpServletRequest) {
requestWrapper = new BodyCopyWrapper((HttpServletRequest) request);
}
if (null == requestWrapper) {
log.warn("未進(jìn)行request包裝返回原來的request");
chain.doFilter(request, response);
} else {
chain.doFilter(requestWrapper, response);
}
}
}
}四、業(yè)務(wù)代碼中使用方式
HttpServletRequest request = ((ServletRequestAttributes) (RequestContextHolder.currentRequestAttributes())).getRequest(); //轉(zhuǎn)對(duì)象 UserLoginDTO userLoginDTO = HttpUtils.getBodyToObject(request, UserLoginDTO.class); //直接獲取內(nèi)容字符串 String bodyStr = HttpUtils.getBodyStr(request);
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- java如何根據(jù)HttpServletRequest獲取IP地址
- Spring?Boot?中正確地在異步線程中使用?HttpServletRequest的方法
- 在 Spring Boot 中使用異步線程時(shí)的 HttpServletRequest 復(fù)用問題記錄
- Java如何獲取HttpServletRequest請(qǐng)求參數(shù)
- 解讀HttpServletRequestWrapper處理request數(shù)據(jù)流多次讀取問題
- 解決子線程中獲取不到HttpServletRequest對(duì)象的問題
- Springboot注入成員變量HttpServletRequest的原理分析
相關(guān)文章
Java 中的 BiFunction 與 BinaryOperator使用示例
在Java8引入的函數(shù)式編程特性中,BiFunction和BinaryOperator是兩個(gè)非常重要的函數(shù)式接口,它們?cè)谔幚矶僮骱秃瘮?shù)組合時(shí)發(fā)揮著關(guān)鍵作用,下面將從基礎(chǔ)概念、核心區(qū)別、實(shí)際應(yīng)用等多個(gè)維度深入解析這兩個(gè)接口,感興趣的朋友跟隨小編一起看看吧2025-09-09
SpringAOP核心對(duì)象的創(chuàng)建圖解
這篇文章主要介紹了SpringAOP核心對(duì)象的創(chuàng)建詳解,通過使用AOP,我們可以將橫切關(guān)注點(diǎn)(如日志記錄、性能監(jiān)控、事務(wù)管理等)從業(yè)務(wù)邏輯中分離出來,使得代碼更加模塊化、可維護(hù)性更高,需要的朋友可以參考下2023-10-10
Java8新特性時(shí)間日期庫(kù)DateTime API及示例詳解
這篇文章主要介紹了Java8新特性時(shí)間日期庫(kù)DateTime API及示例詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10
Mybatis?List列表In查詢實(shí)現(xiàn)的注意事項(xiàng)說明
這篇文章主要介紹了Mybatis?List列表In查詢實(shí)現(xiàn)的注意事項(xiàng)說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02
Java Stream 的 flatMap 與 map 的核
map進(jìn)行元素到元素的單層轉(zhuǎn)換,flatMap則將元素映射為流后再扁平化處理,適用于嵌套結(jié)構(gòu)展開,二者核心差異在于是否展開多層數(shù)據(jù),選擇時(shí)需根據(jù)數(shù)據(jù)結(jié)構(gòu)層級(jí)和性能需求決定,本文給大家介紹Java Stream 的flatMap與map的核心區(qū)別,感興趣的朋友一起看看吧2025-08-08
Eclipse 2020-06 漢化包安裝步驟詳解(附漢化包+安裝教程)
這篇文章主要介紹了Eclipse 2020-06 漢化包安裝步驟(附漢化包+安裝教程),本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08
Java利用位運(yùn)算實(shí)現(xiàn)加減運(yùn)算詳解
這篇文章主要為大家介紹了如何使用位運(yùn)算來實(shí)現(xiàn)加減功能,也就是在整個(gè)運(yùn)算過程中不能出現(xiàn)加減符號(hào)。文中的示例代碼講解詳細(xì),感興趣的可以了解一下2022-12-12

