解決java攔截器獲取POST入?yún)?dǎo)致@RequestBody參數(shù)丟失問(wèn)題
java攔截器獲取POST入?yún)?dǎo)致@RequestBody參數(shù)丟失
在java開(kāi)發(fā)中,攔截器的使用還是比較廣泛的,例如:限制ip、接口校驗(yàn)、用戶權(quán)限校驗(yàn)等等,大多首選都是使用攔截器,在正式請(qǐng)求接口之前先做一層校驗(yàn),保證接口服務(wù)一定程度上的安全和可靠性
但是小冰在使用攔截器獲取json入?yún)⒌臅r(shí)候,遇到一個(gè)問(wèn)題,那就是攔截器獲取一般都是用流的方式獲取,一般都是能獲取到的,但是會(huì)引發(fā)另外一個(gè)問(wèn)題:
攔截器相關(guān)校驗(yàn)走完之后,接口層的@RequestBody修飾的對(duì)象參數(shù)都會(huì)失效,并且調(diào)試之后釋放會(huì)報(bào)錯(cuò)
org.springframework.http.converter.HttpMessageNotReadableException: I/O error while reading input message; nested exception is java.io.IOException: Stream closed
因?yàn)樾”昧肆?,并且存在流關(guān)閉,一次請(qǐng)求流關(guān)閉了就只能讀取一次,所以到controller接口層就會(huì)報(bào)錯(cuò),流的這個(gè)報(bào)錯(cuò)的意思就是:大哥,我都關(guān)門(mén)了,你還來(lái)找我干嘛,滾!
解決方案
于是找度娘,搜了一波,發(fā)現(xiàn)大多數(shù)處理方案都是相關(guān)報(bào)錯(cuò),于是總結(jié)了一個(gè)小冰解決了問(wèn)題的方案,如下:
- 思路:不管流關(guān)沒(méi)關(guān)閉,要把流中的參數(shù),延伸到后面的接口去用就可以了
- 需要的組織:過(guò)濾器大哥、攔截器大哥、防流失大哥、過(guò)濾器Bean大哥還有一個(gè)自定義方法小弟
- 立馬上車出發(fā):
首先上車的是:自定義方法小弟(用來(lái)獲取參數(shù)-HttpRequestHelper)
/** * @description: 獲取流并將想要的結(jié)果通過(guò)拼接返回 **/ public class HttpRequestHelper { public static String getBodyString(HttpServletRequest request) throws IOException { StringBuilder sb = new StringBuilder(); InputStream inputStream = null; BufferedReader reader = null; try { inputStream = request.getInputStream(); reader = new BufferedReader(new InputStreamReader(inputStream, Charset.forName("UTF-8"))); String line = ""; while ((line = reader.readLine()) != null) { sb.append(line); } } catch (IOException e) { e.printStackTrace(); } finally { if (inputStream != null) { try { inputStream.close(); } catch (IOException e) { e.printStackTrace(); } } if (reader != null) { try { reader.close(); } catch (IOException e) { e.printStackTrace(); } } } return sb.toString(); } }
其次是:防流失大哥(防止流讀取完之后就丟失了-MyRequestWrapper)
/** * @description: 防止流丟失 **/ public class MyRequestWrapper extends HttpServletRequestWrapper { private final byte[] body; public MyRequestWrapper(HttpServletRequest request) throws IOException { super(request); //返回參數(shù)字節(jié)數(shù)組 body = HttpRequestHelper.getBodyString(request).getBytes(Charset.forName("UTF-8")); } @Override public BufferedReader getReader() throws IOException { return new BufferedReader(new InputStreamReader(getInputStream())); } @Override public ServletInputStream getInputStream() throws IOException { final ByteArrayInputStream bais = new ByteArrayInputStream(body); return new ServletInputStream() { @Override public int read() throws IOException { return bais.read(); } @Override public boolean isFinished() { return false; } @Override public boolean isReady() { return false; } @Override public void setReadListener(ReadListener readListener) { } }; } }
然后是:過(guò)濾器大哥(將流中數(shù)據(jù)續(xù)傳-HttpServletRequestFilter)
/** * @description: 使用過(guò)濾器處理流,將當(dāng)前流放到一個(gè)新的request對(duì)象中 **/ public class HttpServletRequestFilter implements Filter { @Override public void destroy() { } @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { ServletRequest requestWrapper = null; if (request instanceof HttpServletRequest) { requestWrapper = new MyRequestWrapper((HttpServletRequest) request); } //獲取請(qǐng)求中的流如何,將取出來(lái)的字符串,再次轉(zhuǎn)換成流,然后把它放入到新request對(duì)象中。 // 在chain.doFiler方法中傳遞新的request對(duì)象 if (requestWrapper == null) { chain.doFilter(request, response); } else { chain.doFilter(requestWrapper, response); } } @Override public void init(FilterConfig arg0) throws ServletException { } }
最后是:攔截器大哥(獲取流數(shù)據(jù)讀取入?yún)?只是代碼片段)
//放到繼承HandlerInterceptor的類下的preHandle方法中指定業(yè)務(wù)位置就行啦,這里讀出來(lái)的string, //這里我處理為一個(gè)json(就是入?yún)?duì)象的各個(gè)字段和字段值),方便操作 //構(gòu)造函數(shù)的httpServletRequest參數(shù)是preHandle方法的request入?yún)? JSONObject.parseObject(HttpRequestHelper.getBodyString(httpServletRequest));
然后運(yùn)行,由于各位大哥就先走了,發(fā)現(xiàn)還是不對(duì)勁,好像丟了個(gè)司機(jī),于是一回頭一看,最牛的司機(jī)大哥還沒(méi)上車,這是哪位大哥?答:過(guò)濾器Bean大哥
司機(jī)大哥:過(guò)濾器Bean大哥(將上面的過(guò)濾器大哥放在副駕上(在springboot的啟動(dòng)類中注入過(guò)濾器))
@Bean public FilterRegistrationBean httpServletRequestReplacedRegistration() { FilterRegistrationBean registration = new FilterRegistrationBean(); //把過(guò)濾器大哥安放到副駕上 registration.setFilter(new HttpServletRequestFilter()); registration.addUrlPatterns("/*"); registration.addInitParameter("paramName", "paramValue"); registration.setName("httpServletRequestFilter"); registration.setOrder(1); return registration; }
然后運(yùn)行,攔截獲取json參數(shù),因?yàn)橛羞@幾個(gè)大哥這么組織和坐鎮(zhèn),一開(kāi)車,沒(méi)問(wèn)題,超速上高速,shua~
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java中常見(jiàn)的XML解析方法與應(yīng)用詳解
XML(eXtensible Markup Language)是一種用于存儲(chǔ)和傳輸數(shù)據(jù)的標(biāo)記語(yǔ)言,被廣泛應(yīng)用于表示和交換獨(dú)立于應(yīng)用程序和硬件平臺(tái)的結(jié)構(gòu)化信息,下面我們就來(lái)看看它的常見(jiàn)解析方法有哪些吧2024-01-01mybatis實(shí)現(xiàn)批量插入并返回主鍵(xml和注解兩種方法)
這篇文章主要介紹了mybatis實(shí)現(xiàn)批量插入并返回主鍵(xml和注解兩種方法),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12mybatisPlus更新策略導(dǎo)致更新失敗問(wèn)題
這篇文章主要介紹了mybatisPlus更新策略導(dǎo)致更新失敗問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08Java實(shí)現(xiàn)開(kāi)箱即用的redis分布式鎖
這篇文章主要為大家詳細(xì)介紹了如何使用Java實(shí)現(xiàn)開(kāi)箱即用的基于redis的分布式鎖,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,需要的可以收藏一下2022-12-12SpringBoot集成Druid實(shí)現(xiàn)多數(shù)據(jù)源的兩種方式
這篇文章主要介紹了SpringBoot集成Druid實(shí)現(xiàn)多數(shù)據(jù)源的兩種方式,集成com.baomidou的方式和基于AOP手動(dòng)實(shí)現(xiàn)多數(shù)據(jù)源原生的方式,文中通過(guò)代碼示例講解的非常詳細(xì),需要的朋友可以參考下2024-03-03SpringBoot2 JPA解決懶加載異常的問(wèn)題
這篇文章主要介紹了SpringBoot2 JPA解決懶加載異常的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-01-01