Java中的异常分为两大类:
try {
// 可能抛出异常的代码
} catch (SpecificException e) {
// 处理特定异常
} catch (GeneralException e) {
// 处理更一般的异常
} finally {
// 无论是否发生异常都会执行的代码
}
@ControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler(Exception.class)
public ResponseEntity<ErrorResponse> handleAllExceptions(Exception ex) {
ErrorResponse error = new ErrorResponse(
"SERVER_ERROR",
"An unexpected error occurred",
HttpStatus.INTERNAL_SERVER_ERROR.value()
);
return new ResponseEntity<>(error, HttpStatus.INTERNAL_SERVER_ERROR);
}
@ExceptionHandler(ResourceNotFoundException.class)
public ResponseEntity<ErrorResponse> handleResourceNotFound(
ResourceNotFoundException ex) {
ErrorResponse error = new ErrorResponse(
"NOT_FOUND",
ex.getMessage(),
HttpStatus.NOT_FOUND.value()
);
return new ResponseEntity<>(error, HttpStatus.NOT_FOUND);
}
}
public class BusinessException extends RuntimeException {
private String errorCode;
private HttpStatus httpStatus;
public BusinessException(String message, String errorCode, HttpStatus httpStatus) {
super(message);
this.errorCode = errorCode;
this.httpStatus = httpStatus;
}
// getters
}
@Data
@AllArgsConstructor
@NoArgsConstructor
public class ErrorResponse {
private String errorCode;
private String message;
private int status;
private long timestamp;
private String path;
private List<ValidationError> validationErrors;
public ErrorResponse(String errorCode, String message, int status) {
this.errorCode = errorCode;
this.message = message;
this.status = status;
this.timestamp = System.currentTimeMillis();
}
}
@Data
@AllArgsConstructor
public class ValidationError {
private String field;
private String message;
}
@ExceptionHandler(MethodArgumentNotValidException.class)
public ResponseEntity<ErrorResponse> handleValidationExceptions(
MethodArgumentNotValidException ex) {
List<ValidationError> errors = ex.getBindingResult()
.getFieldErrors()
.stream()
.map(error -> new ValidationError(
error.getField(),
error.getDefaultMessage()))
.collect(Collectors.toList());
ErrorResponse error = new ErrorResponse(
"VALIDATION_FAILED",
"Validation failed for one or more fields",
HttpStatus.BAD_REQUEST.value(),
errors
);
return new ResponseEntity<>(error, HttpStatus.BAD_REQUEST);
}
@ExceptionHandler(BusinessException.class)
public ResponseEntity<ErrorResponse> handleBusinessException(
BusinessException ex,
WebRequest request,
Locale locale) {
String message = messageSource.getMessage(
ex.getMessageKey(),
ex.getArgs(),
locale);
ErrorResponse error = new ErrorResponse(
ex.getErrorCode(),
message,
ex.getHttpStatus().value()
);
return new ResponseEntity<>(error, ex.getHttpStatus());
}
@ExceptionHandler(Exception.class)
public ResponseEntity<ErrorResponse> handleAllExceptions(Exception ex, WebRequest request) {
logger.error("Unexpected error occurred: {}", ex.getMessage(), ex);
ErrorResponse error = new ErrorResponse(
"SERVER_ERROR",
"An unexpected error occurred",
HttpStatus.INTERNAL_SERVER_ERROR.value(),
request.getDescription(false)
);
return new ResponseEntity<>(error, HttpStatus.INTERNAL_SERVER_ERROR);
}
分层异常处理:
异常转换:将底层异常转换为上层业务异常
异常信息设计:
监控与报警:对关键异常设置监控报警
文档化:为API消费者提供清晰的错误代码和含义文档
@ControllerAdvice
public class CustomErrorController implements ErrorController {
@RequestMapping("/error")
public ResponseEntity<ErrorResponse> handleError(HttpServletRequest request) {
Integer status = (Integer) request.getAttribute("javax.servlet.error.status_code");
Exception exception = (Exception) request.getAttribute("javax.servlet.error.exception");
ErrorResponse error = new ErrorResponse(
"NOT_FOUND",
"Requested resource not found",
status
);
return new ResponseEntity<>(error, HttpStatus.valueOf(status));
}
}
@ExceptionHandler
public ResponseEntity<ErrorResponse> handleCorsException(CorsException ex) {
ErrorResponse error = new ErrorResponse(
"CORS_ERROR",
ex.getMessage(),
HttpStatus.FORBIDDEN.value()
);
return new ResponseEntity<>(error, HttpStatus.FORBIDDEN);
}
@ExceptionHandler(MaxUploadSizeExceededException.class)
public ResponseEntity<ErrorResponse> handleMaxSizeException(
MaxUploadSizeExceededException exc) {
ErrorResponse error = new ErrorResponse(
"FILE_TOO_LARGE",
"File size exceeds the allowed limit",
HttpStatus.BAD_REQUEST.value()
);
return new ResponseEntity<>(error, HttpStatus.BAD_REQUEST);
}
良好的异常处理是构建健壮Java应用程序的关键。通过合理使用Java原生异常机制和Spring的全局异常处理功能,可以:
记住:异常处理的目标不是消灭所有异常,而是以可控的方式处理异常,并提供有意义的反馈。