如何優雅的處理異常?SpringBoot全域性異常與資料校驗
要思考的問題
在現在的前後端互動中,通常都規範了介面返回方式,如返回的介面狀態(成功|失敗)以及要返回的資料在那個欄位取,或者說失敗了以後提示資訊從介面哪裡返回,因此,如果想做全域性異常,並且異常發生後能準確的返回給前端解析,那麼需要異常發生時返回給前端的格式與正常失敗場景的格式一致。
專案建立
利用idea 工具,很容易的搭建一個SpringBoot專案,要引入的maven依賴如下:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-validation</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency>
很簡單,除了加入web功能還加入了我們需要用到的JSR-303校驗框架。
定義成功失敗 返回碼
public class Code { /** * 成功 */ public static int SUCCESSED = 1; /** * 失敗 */ public static int FAILED = -1; }
定義介面返回響應實體
public class Response<T> implements Serializable{ /** * */ private static final long serialVersionUID = 4250719891313555820L; /** * 返回結果集 */ private T result; /** * 返回訊息 */ private String msg; /** * 響應碼 */ private Integer code; //set get 略 }
全域性異常攔截和驗證
定義自定義業務異常
public class MyException extends RuntimeException { private static final long serialVersionUID = -5875371379845226068L; public MyException(){} public MyException(String msg){ this.msg = msg ; } /** * 異常資訊 */ private String msg ; /** * 具體異常碼 */ private int code = Code.FAILED; get set 略
編寫全域性異常控制器並對自定義異常做處理
@ControllerAdvice public class GlobalExceptionHandler { private Logger logger = LoggerFactory.getLogger(GlobalExceptionHandler.class); @ExceptionHandler(value = MyException.class) @ResponseBody public Response<String> myExceptionErrorHandler(MyException ex) throws Exception { logger.error("myExceptionErrorHandler info:{}",ex.getMessage()); Response<String> r = new Response<>(); r.setMsg(ex.getMsg()); r.setCode(ex.getCode()); return r; }
編寫controller模擬丟擲業務異常
@RestController @RequestMapping("/user") public class UserController { @PostMapping(value = "/update") Response<Boolean> update(User user){ //todo 此處為模擬異常丟擲 if(true){ throw new MyException("更新失敗"); } //todo 此處為模擬返回 Response<Boolean> response = new Response<>(); response.setCode(Code.SUCCESSED); response.setResult(true); return response; } }
postMan模擬請求介面,進行驗證
資料繫結異常處理
通常我們操作資料的時候,不僅前端需要進行資料校驗,後端也應當進行攔截和進行相應的錯誤提示,JSR-303校驗框架也是我們的一種選擇。
編寫實體`User`,並對屬性進行註解控制
public class User { @NotNull(message = "使用者名稱不能為空") private String userName; private int age; //...
全域性異常控制類加入攔截
@ControllerAdvice public class GlobalExceptionHandler { private Logger logger = LoggerFactory.getLogger(GlobalExceptionHandler.class); @ExceptionHandler(value = BindException.class) @ResponseBody public Response<String> bindExceptionErrorHandler(BindException ex) throws Exception { logger.error("bindExceptionErrorHandler info:{}",ex.getMessage()); Response<String> r = new Response<>(); StringBuilder sb = new StringBuilder(); FieldError fieldError = ex.getFieldError(); sb.append(fieldError.getDefaultMessage()); r.setMsg(sb.toString()); r.setCode(Code.FAILED); return r; } //...
編寫控制器
@RestController @RequestMapping("/user") public class UserController { @PostMapping(value = "/add") Response<User> add(@Validated User user){ //todo 此處為模擬返回 Response<User> response = new Response<>(); response.setCode(Code.SUCCESSED); response.setResult(new User()); return response; } //...
postMan模擬請求
不填寫任何屬性,模擬新增操作,準確進行攔截和報錯
專案結構預覽:
結尾
適合的才是最好的,每個團隊都應摸索出自己的一套異常解決方案,本文所提僅針對業務異常,希望大家也能有所收穫
【責任編輯:龐桂玉 TEL:(010)68476606】