java - Spring boot父(2.3.0.RELEASE)属性为null时不调用异常处理方法

标签 java spring-boot spring-restcontroller

我有一个剩余 Controller ,它有一个 requestObject,其中包含 3 个属性,并且它们是强制性的。基本上我有下面的类,其中包含方法constraintViolationException来处理@NotBlank异常。

@ControllerAdvice
public class MyExceptionHandler extends ResponseEntityExceptionHandler{
    // @Validate For Validating Path Variables and Request Parameters
    @ExceptionHandler(ConstraintViolationException.class)
    public void constraintViolationException(HttpServletResponse response) throws IOException {
  //TODO - to provide custom error message
        response.sendError(HttpStatus.BAD_REQUEST.value());
    }

}

我的 Controller 如下所示:

@RestController
public class MyRestController {



    @PostMapping(value = "/processMyPayment")
    public  ResponseEntity<?> processMyPayment(@Valid MyRequest myRequest ) {

        MyResponse myResponse = new MyResponse();
        return new ResponseEntity<>(myTResponse, HttpStatus.OK);
    }
...

当我添加类 MyExceptionHandler 并将 null 属性发送给我的其余 Controller 时,它应该调用constraintViolationException,因为值是 null,但事实并非如此,我得到以下结果: enter image description here

。但是,当我删除 extends ResponseEntityExceptionHandler 时,我得到了整个堆栈跟踪:

"timestamp": "2020-06-05T13:01:47.715+00:00",
"status": 400,
"error": "Bad Request",
"trace": "org.springframework.validation.BindException: org.springframework.validation.BeanPropertyBindingResult: 3 errors\nField error in object 'myRequest' on field 'amount': rejected value [null]; codes [NotBlank.myRequest.amount,NotBlank.amount,NotBlank.java.lang.String,NotBlank]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [myRequest.amount,amount]; arguments []; default message [amount]]; default message [Amount name mandatory]\nField error in object 'myRequest' on field 'accountId': rejected value [null]; codes [NotBlank.myRequest.accountId,NotBlank.accountId,NotBlank.java.lang.String,NotBlank]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [myRequest.accountId,accountId]; arguments []; default message [accountId]]; default message [Account id is mandatory]\nField error in object 'myRequest' on field 'accountName': rejected value [null]; codes [NotBlank.myRequest.accountName,NotBlank.accountName,NotBlank.java.lang.String,NotBlank]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [myRequest.accountName,accountName]; arguments []; default message [accountName]]; default message [Account name mandatory]\r\n\tat org.springframework.web.method.annotation.ModelAttributeMethodProcessor.resolveArgument(ModelAttributeMethodProcessor.java:164)\r\n\tat org.springframework.web.method.support.HandlerMethodArgumentResolverComposite.resolveArgument(HandlerMethodArgumentResolverComposite.java:121)\r\n\tat org.springframework.web.method.support.InvocableHandlerMethod.getMethodArgumentValues(InvocableHandlerMethod.java:167)\r\n\tat org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:134)\r\n\tat org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:105)\r\n\tat org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:879)\r\n\tat org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:793)\r\n\tat org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87)\r\n\tat org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1040)\r\n\tat org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:943)\r\n\tat org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006)\r\n\tat org.springframework.web.servlet.FrameworkServlet.doPost(FrameworkServlet.java:909)\r\n\tat javax.servlet.http.HttpServlet.service(HttpServlet.java:660)\r\n\tat org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883)\r\n\tat javax.servlet.http.HttpServlet.service(HttpServlet.java:741)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:231)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)\r\n\tat org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)\r\n\tat org.springframework.web.filter.RequestContextFilter.doFilterInternal(RequestContextFilter.java:100)\r\n\tat org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)\r\n\tat org.springframework.web.filter.FormContentFilter.doFilterInternal(FormContentFilter.java:93)\r\n\tat org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)\r\n\tat org.springframework.boot.actuate.metrics.web.servlet.WebMvcMetricsFilter.doFilterInternal(WebMvcMetricsFilter.java:93)\r\n\tat org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)\r\n\tat org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:201)\r\n\tat org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)\r\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)\r\n\tat org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:202)\r\n\tat org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:96)\r\n\tat org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:541)\r\n\tat org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:139)\r\n\tat org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:92)\r\n\tat org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:74)\r\n\tat org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:343)\r\n\tat org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:373)\r\n\tat org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:65)\r\n\tat org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:868)\r\n\tat org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1590)\r\n\tat org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49)\r\n\tat java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1130)\r\n\tat java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:630)\r\n\tat org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61)\r\n\tat java.base/java.lang.Thread.run(Thread.java:832)\r\n",
"message": "Validation failed for object='myRequest'. Error count: 3",
"errors":[
{"codes":["NotBlank.myRequest.amount", "NotBlank.amount", "NotBlank.java.lang.String", "NotBlank" ],…},
{"codes":["NotBlank.myRequest.accountId", "NotBlank.accountId", "NotBlank.java.lang.String", "NotBlank" ],…},
{"codes":["NotBlank.myRequest.accountName", "NotBlank.accountName", "NotBlank.java.lang.String", "NotBlank" ],…}
],
"path": "/processMyPayment"
}

基本上,我想要实现的是,我不想提供堆栈跟踪,而是想添加为空的字段,并带有一条错误消息,表明这些字段不能为空。

知道如何实现这一目标吗? 提前致谢。

最佳答案

试试这个:

/**
     * Handles javax.validation.ConstraintViolationException. Thrown when @Validated fails.
     *
     * @param ex the ConstraintViolationException
     * @return error
     */
    @ExceptionHandler(javax.validation.ConstraintViolationException.class)
    public ResponseEntity<Object> handleConstraintViolation(
            javax.validation.ConstraintViolationException ex) {
        // list of violations
        // ex.getConstraintViolations();

        // return custom error
        return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(null);
    }

您应该重写方法 handleMethodArgumentNotValid()ResponseEntityExceptionHandler ,像这样:

/**
     * Handle @see MethodArgumentNotValidException. Triggered when an object fails validation.
     *
     * @param ex      the @see MethodArgumentNotValidException that is thrown when validation fails
     * @param headers   HttpHeaders @see HttpHeaders
     * @param status    HttpStatus  @see HttpStatus
     * @param request   WebRequest @see WebRequest
     * @return ResponseEntity<Object>
     */
    @Override
    protected ResponseEntity<Object> handleMethodArgumentNotValid(
            MethodArgumentNotValidException ex,
            HttpHeaders headers,
            HttpStatus status,
            WebRequest request) {

            // get Binding Result
            Object bindingResult = ex.getBindingResult();
            // get specific field errors
            Object fieldErrors = ex.getBindingResult().getFieldErrors();
            // get error counts
            int errorCount = ex.getBindingResult().getErrorCount();

            // return, you can create custom responses based on fieldErrors and errorCounts as well and send that in the body below.
            return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(fieldErrors);
    }

当你按照上面的方法重写这个函数时,你会发现只要任何 @Valid 就会调用这个方法。验证失败。您可以更改返回类型。

测试后告诉我。您还可以检查开源测试项目的自定义异常处理程序和响应 -> https://github.com/AccessGateLabs/response-builder/

处理自定义异常的特定类的链接 -> https://github.com/AccessGateLabs/response-builder/blob/master/src/main/java/com/accessgatelabs/oss/builder/exceptions/RestExceptionHandler.java

关于java - Spring boot父(2.3.0.RELEASE)属性为null时不调用异常处理方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62216419/

相关文章:

java - HashMap返回方法

spring-boot - Spring Security - 当获取登录页面时,安全性尝试进行身份验证并返回 401 错误

java - 将管理基本路径设置为 "/",并将 server.context-path 设置为 myApp

java - Spring 中的内容类型和@ResponseBody

json - 在 POST 中没有将属性设置为 spring @RestController

java - 将 CompletableFuture 与 EJb 结合使用时遇到困难

java - 是否可以将定位器从属性文件传递到 POM 中的 @FindBy 注释?

Java For循环三角形图案

java - 将日期传递给 postman 时出现“未找到转换器”错误

java - 在 Spring REST Controller 中使用 Pageable 防止多个排序参数