日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

SpringBoot如何使用RequestBodyAdvice進行統一參數處理

瀏覽:215日期:2023-02-28 17:04:09
SpringBoot RequestBodyAdvice參數處理

在實際項目中 , 往往需要對請求參數做一些統一的操作 , 例如參數的過濾 , 字符的編碼 , 第三方的解密等等 , Spring提供了RequestBodyAdvice一個全局的解決方案 , 免去了我們在Controller處理的繁瑣 .

RequestBodyAdvice僅對使用了@RqestBody注解的生效 , 因為它原理上還是AOP , 所以GET方法是不會操作的.

package com.xbz.common.web; import org.springframework.core.MethodParameter;import org.springframework.http.HttpHeaders;import org.springframework.http.HttpInputMessage;import org.springframework.http.converter.HttpMessageConverter;import org.springframework.web.bind.annotation.ControllerAdvice;import org.springframework.web.servlet.mvc.method.annotation.RequestBodyAdvice; import java.io.IOException;import java.io.InputStream;import java.lang.reflect.Type; /** * @title 全局請求參數處理類 * @author Xingbz * @createDate 2019-8-2 */@ControllerAdvice(basePackages = 'com.xbz.controller')//此處設置需要當前Advice執行的域 , 省略默認全局生效public class GlobalRequestBodyAdvice implements RequestBodyAdvice { /** 此處如果返回false , 則不執行當前Advice的業務 */ @Override public boolean supports(MethodParameter methodParameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {//return methodParameter.getMethod().isAnnotationPresent(XXApiReq.class);return false; } /** * @title 讀取參數前執行 * @description 在此做些編碼 / 解密 / 封裝參數為對象的操作 * * */ @Override public HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) throws IOException {return new XHttpInputMessage(inputMessage, 'UTF-8'); } /** * @title 讀取參數后執行 * @author Xingbz */ @Override public Object afterBodyRead(Object body, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {return inputMessage; } /** * @title 無請求時的處理 */ @Override public Object handleEmptyBody(Object body, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {return body; }} //這里實現了HttpInputMessage 封裝一個自己的HttpInputMessageclass XHttpInputMessage implements HttpInputMessage { private HttpHeaders headers; private InputStream body; public XHttpInputMessage(HttpInputMessage httpInputMessage, String encode) throws IOException {this.headers = httpInputMessage.getHeaders();this.body = encode(httpInputMessage.getBody(), encode); } private InputStream encode(InputStream body, String encode) {//省略對流進行編碼的操作return body; } @Override public InputStream getBody() {return body; } @Override public HttpHeaders getHeaders() {return null; }}

Spring默認提供了接口的抽象實現類RequestBodyAdviceAdapter , 我們可以繼承這個類按需實現 , 讓代碼更簡潔一點

package org.springframework.web.servlet.mvc.method.annotation; import java.io.IOException;import java.lang.reflect.Type; import org.springframework.core.MethodParameter;import org.springframework.http.HttpInputMessage;import org.springframework.http.converter.HttpMessageConverter;import org.springframework.lang.Nullable; public abstract class RequestBodyAdviceAdapter implements RequestBodyAdvice { @Overridepublic HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter,Type targetType, Class<? extends HttpMessageConverter<?>> converterType)throws IOException { return inputMessage;} @Overridepublic Object afterBodyRead(Object body, HttpInputMessage inputMessage, MethodParameter parameter,Type targetType, Class<? extends HttpMessageConverter<?>> converterType) { return body;} @Override@Nullablepublic Object handleEmptyBody(@Nullable Object body, HttpInputMessage inputMessage,MethodParameter parameter, Type targetType,Class<? extends HttpMessageConverter<?>> converterType) { return body;}}Springboot 對RequestBody的值進行統一修改的幾種方式背景

最近在項目中遇到需要統一對Request請求中的某一個自定義對象的屬性進行統一修改的需求。

考慮了幾種實現方式,現在記錄一下。由于原項目過于復雜,自己寫幾個demo進行記錄。

解決方式

SpringBoot如何使用RequestBodyAdvice進行統一參數處理

方式一:利用filter進行處理

大坑:

​ 如果你想要改變加了RequestBody注解的數據,無論如何你都要通過getInputStream()方法來獲取流來拿到對應的參數,然后更改。在不經過拿取流的情況下,spring的RequestBody注解也是通過getInputStream()方法來獲取流來映射為request對象。

但是如果你想要的統一的進行修改,也必須經過getInputStream()來首先拿到stream然后才能進行修改。但此時stream被消費之后,就會關閉。

然后你的controller中的參數就拿不到對象,報錯如下。

I/O error while reading input message; nested exception is java.io.IOException: Stream closed

可以通過創建并使用自定義的的HttpServletRequestWrapper來避免這種情況。

步驟一:編寫自定義HttpServletRequestWrapper

package com.example.testlhf.filter;import com.alibaba.fastjson.JSON;import com.alibaba.fastjson.JSONObject;import com.example.testlhf.entity.Student;import lombok.extern.slf4j.Slf4j;import javax.servlet.ReadListener;import javax.servlet.ServletInputStream;import javax.servlet.ServletRequest;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.InputStream;import java.io.InputStreamReader;import java.nio.charset.Charset;/** * @Description TODO * @Author yyf * @Date 2020/10/29 12:48 * @Version 1.0 **/@Slf4jpublic class ChangeStudentNameRequestWrapper extends HttpServletRequestWrapper { /** * 存儲body數據的容器 */ private byte[] body; public ChangeStudentNameRequestWrapper(HttpServletRequest request) throws IOException {super(request);//接下來的request使用這個String bodyStr = getBodyString(request);body = bodyStr.getBytes(Charset.defaultCharset()); } /** * 獲取請求Body * * @param request request * @return String */ public String getBodyString(final ServletRequest request) {try { return inputStream2String(request.getInputStream());} catch (IOException e) { log.error('', e); throw new RuntimeException(e);} } /** * 獲取請求Body * * @return String */ public String getBodyString() {final InputStream inputStream = new ByteArrayInputStream(body);return inputStream2String(inputStream); } /** * 將inputStream里的數據讀取出來并轉換成字符串 * * @param inputStream inputStream * @return String */ private String inputStream2String(InputStream inputStream) {StringBuilder sb = new StringBuilder();BufferedReader reader = null;try { reader = new BufferedReader(new InputStreamReader(inputStream, Charset.defaultCharset())); String line; while ((line = reader.readLine()) != null) {sb.append(line); }} catch (IOException e) { log.error('', e); throw new RuntimeException(e);} finally { if (reader != null) {try { reader.close();} catch (IOException e) { log.error('', e);} }}JSONObject jsonObject = JSONObject.parseObject(sb.toString());if (jsonObject != null && jsonObject.get('student') != null) { Student student = JSON.toJavaObject((JSON) jsonObject.get('student'), Student.class); log.info('修改之前的學生名稱為:' + student.getName()); student.setName('amd'); jsonObject.put('student', student); return jsonObject.toJSONString();}return sb.toString(); } @Override public BufferedReader getReader() throws IOException {return new BufferedReader(new InputStreamReader(getInputStream())); } @Override public ServletInputStream getInputStream() throws IOException {final ByteArrayInputStream inputStream = new ByteArrayInputStream(body);return new ServletInputStream() { @Override public int read() throws IOException {return inputStream.read(); } @Override public boolean isFinished() {return false; } @Override public boolean isReady() {return false; } @Override public void setReadListener(ReadListener readListener) { }}; }}

步驟二:使用自定義的HttpServletRequestWrapper取代原有的

使用自定義的request取代原有的傳遞給過濾器鏈。

package com.example.testlhf.filter;import lombok.extern.slf4j.Slf4j;import javax.servlet.Filter;import javax.servlet.FilterChain;import javax.servlet.FilterConfig;import javax.servlet.ServletException;import javax.servlet.ServletRequest;import javax.servlet.ServletResponse;import javax.servlet.http.HttpServletRequest;import java.io.IOException;/** * @Description TODO * @Author yyf * @Date 2020/10/29 13:20 * @Version 1.0 **/@Slf4jpublic class ReplaceStreamFilter implements Filter { @Override public void init(FilterConfig filterConfig) throws ServletException {log.info('StreamFilter初始化...'); } @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {ServletRequest requestWrapper = null; //獲取請求中的流,將取出來的字符串,再次轉換成流,然后把它放入到新request對象中,if (request instanceof HttpServletRequest) { requestWrapper = new ChangeStudentNameRequestWrapper((HttpServletRequest) request);}// 在chain.doFiler方法中傳遞新的request對象if (requestWrapper == null) { chain.doFilter(request, response);} else { chain.doFilter(requestWrapper, response);} } @Override public void destroy() {log.info('StreamFilter銷毀...'); }}

步驟三:將過濾器注冊進spring容器

package com.example.testlhf.filter;import org.springframework.boot.web.servlet.FilterRegistrationBean;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import javax.servlet.Filter;/** * @Description TODO * @Author yyf * @Date 2020/10/29 14:20 * @Version 1.0 **/@Configurationpublic class MyFilterConfig { /** * 注冊過濾器 * * @return FilterRegistrationBean */ @Bean public FilterRegistrationBean someFilterRegistration() {FilterRegistrationBean<Filter> registration = new FilterRegistrationBean<>();registration.setFilter(replaceStreamFilter());registration.addUrlPatterns('/*');registration.setName('replaceStreamFilter');return registration; } /** * 實例化StreamFilter * * @return Filter */ @Bean(name = 'replaceStreamFilter') public Filter replaceStreamFilter() {return new ReplaceStreamFilter(); }}

看下效果:

SpringBoot如何使用RequestBodyAdvice進行統一參數處理

到此使用過濾器對post請求中的參數的修改已經完畢。

方式二:使用攔截器進行處理

當我自以為可以使用攔截器前置通知進行處理時才發現,事情并不簡單。

步驟一:自定義一個攔截器

如下圖實現一個攔截器,preHandle中有HttpServletRequest request參數,雖然可以通過它的流獲取到body中數據,但是如果將body中數據進行修改的話,其并不能傳遞給controller。因為request只有兩個set方法。如果將要統一修改的值攝入Attribute,則還仍需從controller中拿到

SpringBoot如何使用RequestBodyAdvice進行統一參數處理

步驟二:在controller中獲取值

SpringBoot如何使用RequestBodyAdvice進行統一參數處理

雖然用這種方式可以在request中添加統一的參數,也可以從每一個controller中獲取值,但仍需要對每一個controller進行代碼修改,顯然這種方式并不是我們需要的。

方式三:使用切面處理

步驟一:引入aspect所需要使用的maven依賴

<dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> <version>5.1.3.RELEASE</version></dependency>

步驟二:編寫自定義的前置通知以及表達

@Component@Aspectpublic class ChangeStudentNameAdvice { @Before('execution(* com.example.testlhf.service.impl.*.*(..))&&args(addStudentRequset)') public void aroundPoints(AddStudentRequset addStudentRequset) {addStudentRequset.getStudent().setName('amd'); }}

注意此處的形參需要和args括號內的字符串保持一致,否則報錯。

注意此處的形參需要和args括號內的字符串保持一致,否則報錯。

步驟三:開啟注解@EnableAspectJAutoProxy

SpringBoot如何使用RequestBodyAdvice進行統一參數處理

總結:

首先說下filter和interceptor的區別:兩者之間的所依賴的環境不一致,filter作為javaWeb三大組件之一,其作用為:攔截請求,以及過濾相應。其依賴于servlet容器。但interceptor依賴于web框架,例如springmvc框架。最常見的面向切面編程AOP所使用的動態代理模式,即是使用攔截器在service方法執行前或者執行后進行一些操作。他們都可以適用于如下的場景:權限檢查,日志記錄,事務管理等等。當然包括,對所有的請求某些參數進行統一的修改。

比較三種方式,方式一和方式二所謂的攔截基本都是基于對http請求的攔截,filter執行在interceptor之前。雖然filter和interceptor都有類似鏈這種概念,但filter可以將request請求修改之后傳遞給后面的filter,就像電路中的串聯,而interceptor的鏈是獨立的,修改其中一個request并不會影響其他的interceptor,類似并聯,不能做到只修改一處其他不用修改的方式。

簡單來說方式一和方式二針對進入controller進行攔截,而后做一些操作。方式三使用的攔截的理念是針對業務方法的,在執行業務方法的前面對參數進行修改,和spring中對事務控制的實現方式類似。

思考:

雖然第一,第三種方式都可以在技術上實現針對某些方法進行統一的參數修改。但是如果將項目當做一個工程來思考的話,不同于日志打印或者事務控制這種非業務邏輯的處理,這種統一修改某些參數來完成一些操作,已嚴重入侵了業務邏輯。

真正的解決方式要么在請求的源頭就做好參數設置,要么通過配置文件在需要使用的地方來進行某些參數的賦值。

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持好吧啦網。

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
免费不卡在线视频| 国产精品国产三级在线观看| 一区二区三区网站| 亚洲免费精品| 久久久久亚洲| 久久久久99| 久久久久久免费视频| 国产 日韩 欧美一区| 荡女精品导航| 98精品视频| 中文一区一区三区高中清不卡免费| 国产福利一区二区三区在线播放| 国产精品丝袜xxxxxxx| 欧美日韩精品一本二本三本| 国产精品99一区二区| 9久re热视频在线精品| 亚洲精品99| 日韩中出av| 欧美黑人巨大videos精品| 久久三级中文| 日韩国产一区二区| 亚洲综合丁香| 亚洲女人av| 免费一级片91| 国产欧美在线| 日韩欧美一区免费| 99国产精品久久久久久久成人热| 国产亚洲精品自拍| 中文字幕一区日韩精品| 国产精品极品在线观看| 日韩不卡免费高清视频| 日本 国产 欧美色综合| 久久爱www成人| 欧美特黄视频| 欧美日韩一区二区三区四区在线观看 | 国产精品久久久免费| 日韩中文首页| 日韩精品免费视频人成| 国内自拍视频一区二区三区| 久久伦理在线| 午夜亚洲福利| 欧美中文一区二区| 欧美日本久久| 亚洲欧美日韩精品一区二区| 欧美国产先锋| 亚洲精品自拍| 日韩欧美三区| 精品黄色一级片| 久久高清国产| 丝袜美腿一区| 精品一区电影| 国产精东传媒成人av电影| 亚洲精品自拍| 免费成人在线影院| 三级在线观看一区二区| 鲁鲁在线中文| 国产精品不卡| 首页国产精品| 亚洲永久av| av综合电影网站| 亚洲1234区| 999精品色在线播放| 亚洲网址在线观看| 日韩.com| 国产精品一级| 日韩精品三级| 亚洲欧美日韩视频二区| 日韩av自拍| 国产成人久久精品一区二区三区| 92国产精品| 美国欧美日韩国产在线播放| 黄色日韩精品| 国产精品一国产精品| 鲁鲁在线中文| 欧美在线网站| 在线一区二区三区视频| 久久不见久久见国语| 亚洲欧美日本日韩| 伊人久久大香线蕉av不卡| 日产精品一区二区| 一区在线免费观看| 国产乱子精品一区二区在线观看| 国产成人a视频高清在线观看| 夜久久久久久| 久久超级碰碰| 一本一本久久| 中文字幕成在线观看| 国产欧美精品久久| 日韩中文字幕亚洲一区二区va在线| 日本不卡免费高清视频在线| 高清久久精品| 免费在线亚洲| 国产精品专区免费| 国产精品99一区二区三区| 中文字幕亚洲影视| 国产亚洲福利| 国产99久久| 老司机精品视频网| 国产三级一区| 日韩手机在线| 视频一区视频二区中文字幕| 黄色日韩精品| 欧美久久久网站| av高清不卡| 在线观看一区| 日本久久成人网| 亚欧成人精品| 亚洲不卡av不卡一区二区| 国产欧美一级| av高清不卡| 久久久久免费av| 欧美肉体xxxx裸体137大胆| 日韩久久精品网| 欧美日韩精品免费观看视完整| 国产福利片在线观看| 亚洲综合色婷婷在线观看| 日本综合视频| 日韩精品影视| 亚洲人成网77777色在线播放| 精品国产午夜肉伦伦影院| 亚洲精品成a人ⅴ香蕉片| 精品中文在线| 久久激情婷婷| 日韩中文字幕av电影| 免费看日韩精品| 国产精品一区二区三区四区在线观看| 97精品国产| 国产精品v日韩精品v欧美精品网站| 日韩一区电影| 日韩不卡一区二区三区| 亚洲天堂黄色| 欧美日韩国产一区二区三区不卡| 国产麻豆一区| 狠狠久久婷婷| 国产日韩欧美一区二区三区 | 国产欧美日韩一级| 国产专区精品| 国产免费成人| 精品成av人一区二区三区| 精品欧美久久| 国产欧美亚洲一区| 亚洲三级网址| 日韩视频在线一区二区三区 | 国产一区日韩| 日韩一区二区三区在线看| 噜噜噜久久亚洲精品国产品小说| 欧美91在线|欧美| 一本一道久久a久久精品蜜桃| 日韩一区二区三免费高清在线观看| 国产成人久久精品麻豆二区| 国产精品av一区二区| 国产精品亚洲产品| 91成人网在线观看| 国产精品激情| 中文字幕亚洲影视| 日韩欧美另类一区二区| 免费在线成人网| 久久国产欧美| 日韩中文影院| 久久一区亚洲| 国产精品videossex久久发布 | 日韩毛片网站| 西西人体一区二区| 国产精品99一区二区三| 亚洲精品一级| 毛片在线网站| 日本va欧美va欧美va精品| 日本一区二区免费高清| 国产精品一区二区av交换| 亚洲久久视频| 婷婷综合成人| 亚洲一区二区毛片| 麻豆网站免费在线观看| 国产精品99久久免费| 国产亚洲精aa在线看| 国产一级成人av| 国内在线观看一区二区三区 | 日韩在线成人| 国产伦精品一区二区三区在线播放| 日韩二区三区在线观看| 精品免费av| 亚洲精品裸体| 国产精品白丝一区二区三区| 久久电影一区| 婷婷综合成人| 粉嫩av一区二区三区四区五区 | 成人片免费看| 99精品在线免费在线观看| 日韩不卡在线观看日韩不卡视频| 免费视频一区二区三区在线观看| 久久精品青草| 国产精品国产一区| 亚洲人成在线网站| 一本大道色婷婷在线| 精品视频久久| 欧美肉体xxxx裸体137大胆| 国产图片一区| 高清不卡亚洲| 中文字幕一区二区精品区| 欧美精品观看|