除非body为空,否则Spring @ExceptionHandler不返回内容

时间:2017-10-09 20:30:12

标签: java spring spring-mvc exception

我使用Spring @ControllerAdvice来处理异常

@ControllerAdvice
public class RestResponseEntityExceptionHandler extends ResponseEntityExceptionHandler {


    @ExceptionHandler(value = { DataIntegrityViolationException.class})
    @ResponseBody
    public ResponseEntity<String> unknownException(Exception ex, WebRequest req) {
        return new ResponseEntity<>(ex.getCause().getMessage(), new HttpHeaders(), HttpStatus.INTERNAL_SERVER_ERROR);

    }

我遇到的问题是,当异常发生时(当我通过swagger发送请求时),我没有得到预期的异常消息,但是:

{"error": "no response from server"}
Response Code : 0
Response Body : No Content

我可以在调试模式中清楚地看到调用@ExceptionHandler注释的方法。

我已经尝试过方法返回类型,@ResponseBody@ResponseStatus注释和一些其他想到的东西,但似乎我只得到一些非空的响应,当我返回没有正文的ResponseEntity,例如

 ResponseEntity.noContent().build()

ResponseEntity.ok().build()

在这种情况下,我会得到正确的http代码和一些标题

请告知我做错了什么

  • Spring version 4.3.9
  • Spring boot version 1.5.4

提前谢谢

UPD

我进行了实验,这是对我有用的解决方案。 它非常接近其中一个答案 - 我会将其标记为已接受

简而言之,我刚刚创建了自己的dto类,用我感兴趣的异常详细信息填充实例并直接返回 我的代码

@ExceptionHandler(value = { DataIntegrityViolationException.class})
@ResponseStatus(code = HttpStatus.INTERNAL_SERVER_ERROR)
@ResponseBody
public ExceptionDetailHolder unknownException(Exception ex, WebRequest req) {
    final Throwable cause = ex.getCause();
    return new ExceptionDetailHolder("Error interacting with the database server",
                                     cause.getClass() + ":" + cause.getMessage(),
                                     cause.getCause().getClass() + ":" + cause.getCause().getMessage()
                                    );
}

@AllArgsConstructor
@NoArgsConstructor
@Getter
@Setter
private class ExceptionDetailHolder {
    private String message;
    private String exceptionMessage;
    private String innerExceptionMessage;
}

结果(也显示了评论者提出的ex.getMessage和ex.getCause()。getMessage()的内容):

{
  "message": "Error interacting with the database server",
  "exceptionMessage": "class org.hibernate.exception.ConstraintViolationException:could not execute statement",
  "innerExceptionMessage": "class com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException:Column 'allow_copay_payments' cannot be null"
}

1 个答案:

答案 0 :(得分:2)

我处理异常的方式如下所示,我找到了特定的异常,然后在这种情况下创建我自己的类对象ValidationErrorDTO,然后填充该类中的必需字段(ValidationErrorDTO):

@ExceptionHandler(HttpMessageNotReadableException.class)
    @ResponseStatus(HttpStatus.BAD_REQUEST)
    @ResponseBody
    public ResponseEntity<ValidationErrorDTO> processValidationIllegalError(HttpMessageNotReadableException ex,
            HandlerMethod handlerMethod, WebRequest webRequest) {

        Throwable throwable = ex.getMostSpecificCause();
        ValidationErrorDTO errorDTO = new ValidationErrorDTO();
        if (throwable instanceof EnumValidationException) {

            EnumValidationException exception = (EnumValidationException) ex.getMostSpecificCause();

            errorDTO.setEnumName(exception.getEnumName());
            errorDTO.setEnumValue(exception.getEnumValue());
            errorDTO.setErrorMessage(exception.getEnumValue() + " is an invalid " + exception.getEnumName());
        }

        return new ResponseEntity<ValidationErrorDTO>(errorDTO, HttpStatus.BAD_REQUEST);
    }