在使用Spring Boot构建Web应用时,异常处理是一个不可或缺的环节。为了确保系统的鲁棒性和提供良好的用户体验,我们可以利用Spring Boot提供的@ControllerAdvice
注解来实现全局异常处理器。全局异常处理器能够集中处理所有控制器中抛出的异常,使得错误信息得以统一捕获、恰当响应,并为前端或API调用者返回友好的反馈。
首先,我们创建一个标记了@ControllerAdvice
的类,该类中的方法将针对所有带有@Controller注解的控制器进行异常拦截和处理。
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
@ControllerAdvice
public class GlobalExceptionHandler {
// 处理NullPointerException
@ExceptionHandler(NullPointerException.class)
public ResponseEntity<ErrorResponse> handleNullPointerException(NullPointerException ex) {
ErrorResponse error = new ErrorResponse(HttpStatus.BAD_REQUEST.value(), "NullPointerException occurred", ex.getMessage());
return new ResponseEntity<>(error, HttpStatus.BAD_REQUEST);
}
// 处理自定义业务异常
@ExceptionHandler(CustomBusinessException.class)
public ResponseEntity<ErrorResponse> handleCustomBusinessException(CustomBusinessException ex) {
ErrorResponse error = new ErrorResponse(ex.getErrorCode(), ex.getErrorMsg(), ex.getMessage());
return new ResponseEntity<>(error, HttpStatus.BAD_REQUEST);
}
// 通用异常处理方法
@ExceptionHandler(Exception.class)
public ResponseEntity<ErrorResponse> handleAllExceptions(Exception ex) {
ErrorResponse error = new ErrorResponse(HttpStatus.INTERNAL_SERVER_ERROR.value(), "服务器内部错误", ex.getMessage());
return new ResponseEntity<>(error, HttpStatus.INTERNAL_SERVER_ERROR);
}
static class ErrorResponse {
private int code;
private String message;
private String detail;
// 构造函数、getter和setter方法...
}
}
上述代码中,GlobalExceptionHandler
通过@ControllerAdvice
注解成为全局异常处理器。其中包含多个带有@ExceptionHandler
的方法,分别对应不同的异常类型。当特定类型的异常发生时,相应的处理方法会被执行,将原始异常转化为结构化的HTTP响应实体。
在ErrorResponse
类中,我们封装了详细的错误信息,包括状态码、错误消息以及可能存在的详细描述。这样无论何时出现异常,客户端都能接收到格式一致且易于理解的错误反馈。
接下来,在某个Controller中模拟抛出异常:
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class ExampleController {
@GetMapping("/throw-npe")
public String throwNullPointerException() {
Object obj = null;
// 模拟抛出NullPointerException
obj.toString();
return "This will never be returned due to exception";
}
@GetMapping("/throw-business-exception")
public String throwCustomBusinessException() {
throw new CustomBusinessException(4001, "Invalid Request", "A custom business error occurred");
}
}
现在,访问/throw-npe
或/throw-business-exception
路径时,原本会看到堆栈信息的地方,已被替换为我们自定义的结构化错误响应:
对于/throw-npe
:
{
"code": 400,
"message": "NullPointerException occurred",
"detail": "null"
}
对于/throw-business-exception
:
{
"code": 4001,
"message": "Invalid Request",
"detail": "A custom business error occurred"
}
分层异常处理:
可以根据异常发生的层级(如DAO、Service、Controller)设计不同层次的异常处理器类,或者细化同一处理器内的异常处理策略。
日志记录:
在全局异常处理器内应添加日志记录语句,以便在后续分析问题时查看完整的上下文信息。
国际化支持:
若项目需要支持多语言环境,那么异常处理响应的内容也应当支持国际化,可以借助i18n工具包获取对应语言的错误提示信息。
总结来说,Spring Boot全局异常处理器是构建高可用、易维护且具有良好用户体验的后端服务的关键组件之一。通过对异常情况的统一管理,不仅能提升系统的稳定性和故障可读性,还能让开发者更高效地排查和修复问题。在您的下一个Spring Boot项目中,尝试采用全局异常处理方式,定会体验到它所带来的诸多益处。