1. 程式人生 > 程式設計 >Spring下Filter過濾器配置全域性異常處理的詳細步驟

Spring下Filter過濾器配置全域性異常處理的詳細步驟

Spring下Filter過濾器配置全域性異常處理

  • Filter中出現的異常,spring的全域性異常處理器是無法捕獲的,所以filter攔截器中出現的異常會直接的拋向瀏覽器,在瀏覽器中顯示500錯誤。
  • 而我當前的專案中,是在Filter中判斷使用者是否有攜帶Token訪問,如果沒有,則丟擲異常,讓其做登入操作。而且異常資訊要處理成json格式返回給前端。這就很尷尬了。

好了廢話說多了,上解決方案:

結局方案:

Filter攔截器中直接丟擲異常資訊

@Component
public class AdminAuthentiationFilter extends OncePerRequestFilter {

  private final String DEFAULTE_URI = "/api/admin/login";
  @Override
  protected void doFilterInternal(HttpServletRequest req,HttpServletResponse resp,FilterChain filterChain) throws ServletException,IOException {

    String admin_token = req.getHeader("admin_token");

    if(StrUtil.isBlank(admin_token) && !req.getRequestURI().equals(DEFAULTE_URI)){
      //在攔截器中直接丟擲一個異常
      throw new LoginException("使用者未登入,請先登入!");
    }
    filterChain.doFilter(req,resp);
  }
}

第一步:在web.xml中配置錯誤頁,用於捕獲500狀態

<!-- 註冊過濾器-->
<filter>
  <filter-name>myFilter</filter-name>
  <filter-class>com.fenkuan.support.filters.AdminAuthentiationFilter</filter-class>
</filter>
<filter-mapping>
  <filter-name>myFilter</filter-name>
  <url-pattern>/*</url-pattern>
</filter-mapping>
<!--捕獲500錯誤狀態-->
<error-page>
  <error-code>500</error-code>
  <location>/500</location>
</error-page>

第二步:編寫一個FilterException自定義異常類

public class FilterException extends RuntimeException{
	private String data;
  public FilterException(String message) {
    super(message);
  }

  public FilterException(String message,String data) {
    super(message,data);
    this.data = data;
  }
   public String getData() {
    return data;
  }
}

第三步:編寫一個用於處理500錯誤的controller

@RestController
public class FilterErrorController {

  @RequestMapping("/500")
  public void filterError(HttpServletRequest req){
    //獲取servlet請求中的異常屬性。該屬性下儲存了確切的錯誤資訊。
    Throwable t = (Throwable) req.getAttribute("javax.servlet.error.exception");
		//建立一個filterException丟擲,該異常會被全域性異常處理類捕獲,並處理。
    throw new FilterException(t.getMessage());
  }
}

第四步:編寫一個捕獲全域性異常的異常處理類

//全域性異常處理類
@RestControllerAdvice
public class ControllerExceptionHandler{

  @ExceptionHandler(FilterException.class)
  @ResponseStatus(HttpStatus.BAD_REQUEST)
  public BadException<?> filterException(FilterException e){
    BadException<Object> objectBadException = handleExceptionObject(e);
    objectBadException.setStatus(HttpStatus.BAD_REQUEST.value());
    objectBadException.setMessage(e.getMessage());
    return objectBadException;
  }
  
  @ExceptionHandler(Exception.class)
  @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
  public BadException<?> responseException(Exception e){
    	//異常兜底處理
    BadException<?> objectBadException = handleExceptionObject(e);
    objectBadException.setStatus(HttpStatus.INTERNAL_SERVER_ERROR.value());
    objectBadException.setMessage(HttpStatus.INTERNAL_SERVER_ERROR.getReasonPhrase());
    return objectBadException;
  }
//  Throwable是Exception的父類,所以可以使用該型別來接受專案中丟擲的所有異常,包括Exception即其子類。
  private <T> BadException<T> handleExceptionObject(Throwable throwable){
    BadException<T> bad = new BadException<>();
    bad.setMessage(throwable.getMessage());
    return bad;
  }
}

BadException類,用於封裝要返會給前端的異常資訊(這裡使用了Lombok工具)

import lombok.Data;

@Data
public class BadException<T> {
  private Integer status;
  private String message;
  private Object data;
  private Object devData;
}

結果:

在這裡插入圖片描述

到此這篇關於Spring下Filter過濾器配置全域性異常處理的詳細步驟的文章就介紹到這了,更多相關Spring全域性異常處理內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!