Java编程高频的“技术点”-01:自定义全局异常处理器

(一)、自定义全局异常处理器

在 Spring Boot 中,自定义校验异常的统一处理通常通过****@RestControllerAdvice(全局异常处理器) 配合 @ExceptionHandler(异常拦截) 来实现。

这样做的好处是,当 @Valid@Validated 校验失败时,我们不需要在每个 Controller 里写重复的 try-catchif-else 逻辑,而是由全局处理器统一捕获,并向前端返回格式一致、语义清晰的 JSON 错误信息。

以下是实现统一异常处理的标准步骤和完整代码示例:

🛠️ 1. 引入必要的依赖

确保你的 pom.xml 中包含了 Spring Boot 的 Web 和 Validation 启动器(Spring Boot 3.x 默认使用 jakarta.validation):

XML 复制代码
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-validation</artifactId>
</dependency>

🌍 2. 创建全局异常处理器(核心代码)

新建一个类(如 GlobalExceptionHandler),使用 @RestControllerAdvice 注解。我们需要重点捕获两种最常见的校验异常:

  • MethodArgumentNotValidException :处理 Controller 层 @RequestBody 参数校验失败(如 POST/PUT 请求的 JSON 数据)。
  • ConstraintViolationException :处理 GET 请求的 @RequestParam@PathVariable 或 Service 层方法参数校验失败。
java 复制代码
import jakarta.validation.ConstraintViolation;
import jakarta.validation.ConstraintViolationException;
import org.springframework.http.HttpStatus;
import org.springframework.validation.FieldError;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseStatus;
import org.springframework.web.bind.annotation.RestControllerAdvice;

import java.util.HashMap;
import java.util.Map;
import java.util.stream.Collectors;

@RestControllerAdvice
public class GlobalExceptionHandler {

    /**
     * 处理 @RequestBody 参数校验失败 (如 POST/PUT 请求)
     * 异常类型:MethodArgumentNotValidException
     */
  // 返回 400 状态码
    @ExceptionHandler(MethodArgumentNotValidException.class)
    @ResponseStatus(HttpStatus.BAD_REQUEST) 
    public Map<String, Object> handleValidationExceptions(
                            MethodArgumentNotValidException ex) {

        Map<String, Object> response = new HashMap<>();
        response.put("code", 400);
        response.put("message", "参数校验失败");

        // 提取所有字段的错误信息,拼接成易读的格式
        String errors = ex.getBindingResult().getFieldErrors().stream()
                .map(fieldError -> fieldError.getField() + ": " + 
                        fieldError.getDefaultMessage())
                .collect(Collectors.joining(", "));
        
        response.put("errors", errors);
        return response;
    }

    /**
     * 处理 @RequestParam / @PathVariable 
     *   或 Service 层方法参数校验失败 (如 GET 请求)
     * 异常类型:ConstraintViolationException
     */
    @ExceptionHandler(ConstraintViolationException.class)
    @ResponseStatus(HttpStatus.BAD_REQUEST)
    public Map<String, Object> handleConstraintViolationException(
                                        ConstraintViolationException ex) {

        Map<String, Object> response = new HashMap<>();
        response.put("code", 400);
        response.put("message", "参数校验失败");

        // 提取异常中的约束违规信息
        String errors = ex.getConstraintViolations().stream()
                .map(ConstraintViolation::getMessage)
                .collect(Collectors.joining(", "));
        
        response.put("errors", errors);
        return response;
    }


    /**
     * 兜底处理:捕获其他未处理的系统异常,防止直接暴露堆栈信息给前端
     */
    @ExceptionHandler(Exception.class)
    @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    public Map<String, Object> handleSystemException(Exception ex) {

        Map<String, Object> response = new HashMap<>();
        response.put("code", 500);
        response.put("message", "系统内部错误,请联系管理员");

        return response;
    }
}

📝 3. 实际效果演示

假设你有一个接收用户注册的 DTO,并且加上了校验规则:

java 复制代码
public class UserRegisterDTO {
    @NotBlank(message = "用户名不能为空")
    private String username;

    @NotBlank(message = "密码不能为空")
    @Size(min = 6, max = 16, 
             message = "密码长度必须在6到16位之间")
    private String password;
    // getter / setter
}

当前端传入了不合法的 JSON 数据(例如 {"username": "", "password": "123"})时,全局异常处理器会自动拦截并返回如下结构化的 JSON:

java 复制代码
{
  "code": 400,
  "message": "参数校验失败",
  "errors": "username: 用户名不能为空, 
            password: 密码长度必须在6到16位之间"
}

💡 最佳实践建议

  1. 统一返回结构 :在实际的企业级项目中,建议将返回的 Map 替换为一个专门的**ApiResponse<T> 泛型类**(包含 code, message, data 字段),这样能让前后端交互的接口规范更加统一。
  2. 国际化支持 :如果你的项目需要支持多语言,可以在 @NotBlank(message = "{user.username.notblank}") 中使用占位符,并配合 Spring Boot 的国际化(i18n)资源文件(如 messages_zh.properties, messages_en.properties)来动态返回不同语言的错误提示。
  3. 避免暴露敏感信息 :在兜底的 Exception 处理中,千万不要直接把 ex.getMessage() 或堆栈信息返回给前端,以免暴露系统内部逻辑或数据库结构。
相关推荐
潜创微科技2 小时前
IT68353:双DP 1.4 + HDMI 2.0 + USB‑C 三合一转 HDMI 2.0 单芯片KVM切换方案
c语言·开发语言
YsyaaabB2 小时前
ACM 模式通用代码模板
java·c++·python·算法
IT界的老黄牛2 小时前
从 MQ 积压追到事件总线:诊断 4K 线程吃光 7G 内存的实战
java·运维·rocketmq
我命由我123452 小时前
C++ - 面向对象 - 析构函数
android·c语言·开发语言·c++·visualstudio·visual studio·android runtime
小旭95272 小时前
商品详情实现与缓存问题(穿透、击穿、雪崩)解决方案
java·数据库·spring boot·后端·缓存
AI视觉网奇3 小时前
blender底部对齐
开发语言·python·blender
苦逼的猿宝3 小时前
基于springboot的课程作业管理系统(源码+论文)
java·毕业设计·springboot·计算机毕业设计
我本楚狂人www3 小时前
Spring 两大核心思想(一):IoC
java·数据库·spring
宠..3 小时前
QComboBox 方法大全
开发语言·qt