SpringBoot
是为了简化Spring
应用的建立、运行、调试、部署等一系列问题而诞生的产物,前端
自动装配的特性让咱们能够更好的关注业务自己而不是外部的XML配置,咱们只需遵循规范,引入相关的依赖就能够轻易的搭建出一个 WEB 工程
java
实际项目开发中,程序每每会发生各式各样的异常状况,特别是身为服务端开发人员的咱们,web
老是不停的编写接口提供给前端调用,分工协做的状况下,避免不了异常的发生,若是直接将错误的信息直接暴露给用户,spring
这样的体验可想而知,且对黑客而言,详细异常信息每每会提供很是大的帮助…浏览器
一个简单的异常请求的接口springboot
@GetMapping("/test1") public String test1() { // 这里只是模拟异常,假设业务处理的时候出现错误了,或者空指针了等等... int i = 10 / 0; return "test1"; }
打开浏览器访问它的时候发现mvc
又或者是用postman
等模拟工具app
若是这接口是给第三方调用或者是本身公司的系统,看到这种错误估计得暴走吧….ide
采用try-catch
的方式,手动捕获异常信息,而后返回对应的结果集,相信不少人都看到过相似的代码(如:封装成Result对象);函数
该方法虽然间接性的解决错误暴露的问题,一样的弊端也很明显,增长了大量的代码量,当异常过多的状况下对应的catch
层愈发的多了起来,
很难管理这些业务异常和错误码之间的匹配,因此最好的方法就是经过简单配置全局掌控….
@GetMapping("/test2") public Map<String, String> test2() { Map<String, String> result = new HashMap<>(16); // TODO 直接捕获全部代码块,而后在 cache try { int i = 10 / 0; result.put("code", "200"); result.put("data", "具体返回的结果集"); } catch (Exception e) { result.put("code", "500"); result.put("message", "请求错误"); } return result; }
经过上面的阅读你们也大体能了解到为啥须要对异常进行全局捕获了,接下来就看看Spring Boot
提供的解决方案
在pom.xml
中添加上spring-boot-starter-web
的依赖便可
<dependencies> <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> </dependencies>
在应用开发过程当中,除系统自身的异常外,不一样业务场景中用到的异常也不同,为了与标题轻松搞定全局异常
更加的贴切,定义个本身的异常,看看如何捕获…
/** * 自定义异常 */ public class CustomException extends RuntimeException { private static final long serialVersionUID = 4564124491192825748L; private int code; public CustomException() { super(); } public CustomException(int code, String message) { super(message); this.setCode(code); } public int getCode() { return code; } public void setCode(int code) { this.code = code; } }
定义返回的异常信息的格式,这样异常信息风格更为统一
/** * 异常信息模板 */ public class ErrorResponseEntity { private int code; private String message; public ErrorResponseEntity(int code, String message) { this.code = code; this.message = message; } // 省略 get/set }
仔细一看是否是和平时正常写的代码没啥区别,不要急,接着看….
import com.winterchen.exception.CustomException; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RestController; @RestController public class ExceptionController { @GetMapping("/test3") public String test3(Integer num) { // TODO 演示须要,实际上参数是否为空经过 @RequestParam(required = true) 就能够控制 if (num == null) { throw new CustomException(400, "num不能为空"); } int i = 10 / num; return "result:" + i; } }
注解概述
@ControllerAdvice
捕获Controller
层抛出的<wiz_tmp_highlight_tag class="cm-searching">异常,若是添加@ResponseBody
返回信息则为JSON
格式。@RestControllerAdvice
至关于@ControllerAdvice
与@ResponseBody
的结合体。@ExceptionHandler
统一处理一种类的<wiz_tmp_highlight_tag class="cm-searching">异常,减小代码重复率,下降复杂度。建立一个GlobalExceptionHandler
类,并添加上@RestControllerAdvice
注解就能够定义出异常通知类了,而后在定义的方法中添加上@ExceptionHandler
便可实现异常的捕捉…
import com.winterchen.exception.CustomException; import com.winterchen.exception.ErrorResponseEntity; import org.springframework.http.HttpHeaders; import org.springframework.http.HttpStatus; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.MethodArgumentNotValidException; import org.springframework.web.bind.annotation.ExceptionHandler; import org.springframework.web.bind.annotation.RestControllerAdvice; import org.springframework.web.context.request.WebRequest; import org.springframework.web.method.annotation.MethodArgumentTypeMismatchException; import org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 全局异常处理 */ @RestControllerAdvice public class GlobalExceptionHandler extends ResponseEntityExceptionHandler { /** * 定义要捕获的异常 能够多个 @ExceptionHandler({}) * @param request request * @param e exception * @param response response * @return 响应结果 */ @ExceptionHandler(CustomException.class) public ErrorResponseEntity customExceptionHandler(HttpServletRequest request, final Exception e, HttpServletResponse response) { response.setStatus(HttpStatus.BAD_REQUEST.value()); CustomException exception = (CustomException) e; return new ErrorResponseEntity(exception.getCode(), exception.getMessage()); } /** * 捕获 RuntimeException 异常 * 若是你以为在一个 exceptionHandler 经过 if (e instanceof xxxException) 太麻烦 * 那么你还能够本身写多个不一样的 exceptionHandler 处理不一样异常 * @param request request * @param e exception * @param response response * @return 响应结果 */ @ExceptionHandler(RuntimeException.class) public ErrorResponseEntity runtimeExceptionHandler(HttpServletRequest request, final Exception e, HttpServletResponse response) { response.setStatus(HttpStatus.BAD_REQUEST.value()); RuntimeException exception = (RuntimeException) e; return new ErrorResponseEntity(400, exception.getMessage()); } /** * 通用的接口映射异常处理方 */ @Override protected ResponseEntity<Object> handleExceptionInternal(Exception ex, Object body, HttpHeaders headers, HttpStatus status, WebRequest request) { if (ex instanceof MethodArgumentNotValidException) { MethodArgumentNotValidException exception = (MethodArgumentNotValidException) ex; return new ResponseEntity<>(new ErrorResponseEntity(status.value(), exception.getBindingResult().getAllErrors().get(0).getDefaultMessage()), status); } if (ex instanceof MethodArgumentTypeMismatchException) { MethodArgumentTypeMismatchException exception = (MethodArgumentTypeMismatchException) ex; logger.error("参数转换失败,方法:" + exception.getParameter().getMethod().getName() + ",参数:" + exception.getName() + ",信息:" + exception.getLocalizedMessage()); return new ResponseEntity<>(new ErrorResponseEntity(status.value(), "参数转换失败"), status); } return new ResponseEntity<>(new ErrorResponseEntity(status.value(), "参数转换失败"), status); } }
import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication public class SpringBootExceptionApplication { public static void main(String[] args) { SpringApplication.run(SpringBootExceptionApplication.class, args); } }
完成准备事项后,启动Chapter17Application
,经过下面的测试结果能够发现,真的是so easy
,代码变得整洁了,扩展性也变好了…
{"code":400,"message":"num不能为空"}
访问http://localhost:8080/test3?num=0
{"code":400,"message":"/ by zero"}
访问http://localhost:8080/test3?num=5
result:2
转载连接:http://blog.battcn.com/2018/06/01/springboot/v2-other-exception/