亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

spring應用中多次讀取http post方法中的流遇到的問題

發布時間:2020-08-28 04:00:58 來源:腳本之家 閱讀:290 作者:三國夢回 欄目:編程語言

一、問題簡述

先說下為啥有這個需求,在基于spring的web應用中,一般會在controller層獲取http方法body中的數據。

方式1:

比如http請求的content-typeapplication/json的情況下,直接用@RequestBody接收。

方式2:

也有像目前我們在做的這個項目,比較原始,是直接手動讀取流。(不要問我為啥這么原始,第一版也不是我寫的。)

@RequestMapping("/XXX.do")
  public void XXX(HttpServletRequest request, HttpServletResponse response) throws IOException {
    JSONObject jsonObject = WebUtils.getParameters(request);
     //業務處理
    ResponseUtil.setResponse(response, MessageFactory.createSuccessMsg());
  }

WebUtils.getParameters如下:

  public static JSONObject getParameters(HttpServletRequest request) throws IOException {
    InputStream is = null;
    is = new BufferedInputStream(request.getInputStream(), BUFFER_SIZE);
    int contentLength = Integer.valueOf(request.getHeader("Content-Length"));
    byte[] bytes = new byte[contentLength];
    int readCount = 0;
    while (readCount < contentLength) {
      readCount += is.read(bytes, readCount, contentLength - readCount);
    }
    String requestJson = new String(bytes, AppConstants.UTF8);
    if (StringUtils.isBlank(requestJson)) {
      return new JSONObject();
    }
    JSONObject jsonObj = JsonUtils.toJSONObject(requestJson);
    return jsonObj;
  }

當然,不管怎么說,都是對流進行讀取。

問題是,假如我想在controller前面加一層aop,aop里面對進入controller層的方法進行日志記錄,記錄方法參數,應該怎么辦呢。

如果是采用了方式1的話,簡單。spring已經幫我們把參數從流里取出來,給我們提供好了,我們拿著打印一下日志即可。

如果是比較悲劇地采用了我們這種方式,參數里只有個httpServletRequest,那就只有自己去讀取流了,然而,在aop中我們把流讀了的話,

在controller層就讀不到了。

畢竟,流只能讀一次啊。

二、怎么一個流讀多次呢

說一千道一萬,流來自哪里,來自

javax.servlet.ServletRequest#getInputStream

所以,我們的思路,是不是可以這樣,定義一個filter,在filter中將request替換為我們自定義的request。

下面標紅的為自定義的request。

/**
 *
 */
package com.ckl.filter;
import com.ckl.utils.BaseWebUtils;
import com.ckl.utils.MultiReadHttpServletRequest;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.core.annotation.Order;
import org.springframework.http.HttpMethod;
import org.springframework.http.MediaType;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
/**
 * Web流多次讀寫過濾器
 *
 * 攔截所有請求,主要是針對第三方提交過來的請求.
 * 為什么要做成可多次讀寫的流,因為可以在aop層打印日志。
 * 但是不影響controller層繼續讀取該流
 *
 * 該filter的原理:https://stackoverflow.com/questions/10210645/http-servlet-request-lose-params-from-post-body-after-read-it-once/17129256#17129256
 * @author ckl
 */
@Order(1)
@WebFilter(filterName = "cacheRequestFilter", urlPatterns = "*.do")
public class CacheRequestFilter implements Filter {
  private static final Logger logger = LoggerFactory.getLogger(CacheRequestFilter.class);
  @Override
  public void init(FilterConfig filterConfig) throws ServletException {
    // TODO Auto-generated method stub
  }
  @Override
  public void doFilter(ServletRequest request, ServletResponse response,
             FilterChain chain) throws IOException, ServletException {
    HttpServletRequest httpServletRequest = (HttpServletRequest) request;
    logger.info("request uri:{}",httpServletRequest.getRequestURI());
    if (BaseWebUtils.isFormPost(httpServletRequest)){
      httpServletRequest = new MultiReadHttpServletRequest(httpServletRequest);
      String parameters = BaseWebUtils.getParameters(httpServletRequest);
      logger.info("CacheRequestFilter receive post req. body is {}", parameters);
    }else if (isPost(httpServletRequest)){
      //文件上傳請求,沒必要緩存請求
      if (request.getContentType().contains(MediaType.MULTIPART_FORM_DATA_VALUE)){
      }else {
        httpServletRequest = new MultiReadHttpServletRequest(httpServletRequest);
        String parameters = BaseWebUtils.getParameters(httpServletRequest);
        logger.info("CacheRequestFilter receive post req. body is {}", parameters);
      }
    }
    chain.doFilter(httpServletRequest, response);
  }
  @Override
  public void destroy() {
    // TODO Auto-generated method stub
  }
  public static boolean isPost(HttpServletRequest request) {
    return HttpMethod.POST.matches(request.getMethod());
  }
}
MultiReadHttpServletRequest.java:
import org.apache.commons.io.IOUtils;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.BufferedReader;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStreamReader;
/**
 * desc:
 * https://stackoverflow.com/questions/10210645/http-servlet-request-lose-params-from-post-body-after-read-it-once/17129256#17129256
 * @author : ckl
 * creat_date: 2018/8/2 0002
 * creat_time: 13:46
 **/
public class MultiReadHttpServletRequest extends HttpServletRequestWrapper {
  private ByteArrayOutputStream cachedBytes;
  public MultiReadHttpServletRequest(HttpServletRequest request) {
    super(request);
    cachedBytes = new ByteArrayOutputStream();
    ServletInputStream inputStream = null;
    try {
      inputStream = super.getInputStream();
      IOUtils.copy(inputStream, cachedBytes);
    } catch (IOException e) {
      e.printStackTrace();
    }
  }
  @Override
  public ServletInputStream getInputStream() throws IOException {
    return new CachedServletInputStream(cachedBytes);
  }
  @Override
  public BufferedReader getReader() throws IOException {
    return new BufferedReader(new InputStreamReader(getInputStream()));
  }
}

在自定義的request中,構造函數中,先把原始流中的數據讀出來,放到ByteArrayOutputStream cachedBytes中。

并且需要重新定義getInputStream方法。

以后每次程序中調用getInputStream方法時,都會從我們的偷梁換柱的request中的cachedBytes字段,new一個InputStream出來。

看上圖紅色部分:

getInputStream我們返回了自定義的CachedServletInputStream類。

那么,接下來是CachedServletInputStream:

package com.ceiec.webservice.utils;
import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
/**
 * An inputstream which reads the cached request body
 */
public class CachedServletInputStream extends ServletInputStream {
  private ByteArrayInputStream input;
  public CachedServletInputStream(ByteArrayOutputStream cachedBytes) {
    // create a new input stream from the cached request body
    byte[] bytes = cachedBytes.toByteArray();
    input = new ByteArrayInputStream(bytes);
  }
  @Override
  public int read() throws IOException {
    return input.read();
  }
  @Override
  public boolean isFinished() {
    return false;
  }
  @Override
  public boolean isReady() {
    return false;
  }
  @Override
  public void setReadListener(ReadListener readListener) {
  }
}

至此。完整的偷梁換柱就結束了。

現在,請再回過頭去,看文章開頭的代碼,標紅的部分。

是不是豁然開朗了?

三、代碼地址

https://github.com/cctvckl/work_util/tree/master/spring-mvc-multiread-post

直接git 下載即可。

這是個單獨的工程,直接eclipse或者idea導入即可。

spring應用中多次讀取http post方法中的流遇到的問題

運行方法:

spring應用中多次讀取http post方法中的流遇到的問題

我這邊講下idea:

直接運行jetty:run這個goal即可。

然后訪問testPost.do即可(下面把curl貼出來,可以自己在接口測試工具里拼裝):

curl -i -X POST \
-H "Content-Type:application/json" \
-d \
'{"id":"32"}
' \
'http://localhost:8080/springmvc-multiread-post/testPost.do'

我這邊演示下效果,可以發現,兩次都讀出來了:

spring應用中多次讀取http post方法中的流遇到的問題

總結

以上所述是小編給大家介紹的spring應用中多次讀取http post方法中的流,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對億速云網站的支持!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

玉溪市| 尼勒克县| 阳城县| 连州市| 乐山市| 萨嘎县| 苗栗市| 晋江市| 航空| 紫阳县| 重庆市| 大悟县| 徐汇区| 宁城县| 绥江县| 广丰县| 会东县| 尚志市| 炉霍县| 温泉县| 全南县| 仪陇县| 岫岩| 清徐县| 杭锦旗| 陈巴尔虎旗| 珲春市| 石狮市| 勃利县| 乌兰察布市| 余姚市| 三河市| 彩票| 桃园县| 临夏市| 德昌县| 肥西县| 鄂温| 荥阳市| 鞍山市| 大关县|