spring web应用的异常处理

针对RESTful返回json的异常处理

@ControllerAdvice和@ExceptionHandler
字面理解
ControllerAdvice:控制增强
ExceptionHandler:异常处理程序
spring有一个类专门处理异常的基类,已经写好了一些默认的异常处理方法,如果需要可以直接继承这个类:ResponseEntityExceptionHandler,如果不能满足需求可以自己重写某些方法(例:方法3)或者直接自己捕获异常(例:方法1/方法2)

@ControllerAdvicej avadoc定义是:

/**
* Indicates the annotated class assists a "Controller".
*
* <p>Serves as a specialization of {@link Component @Component}, allowing for
* implementation classes to be autodetected through classpath scanning.
*
* <p>It is typically used to define {@link ExceptionHandler @ExceptionHandler},
* {@link InitBinder @InitBinder}, and {@link ModelAttribute @ModelAttribute}
* methods that apply to all {@link RequestMapping @RequestMapping} methods.
*
* @author Rossen Stoyanchev
* @since 3.2
*/

即把@ControllerAdvice注解内部使用@ExceptionHandler、@InitBinder、@ModelAttribute注解的方法应用到所有的 @RequestMapping注解的方法。
理解:
1. @ControllerAdvice:是必须加的
2. @ExceptionHandler:是指定捕获的异常类型的(可以不指定类型)
3. 这是开始执行@RequestMapping方法时发生的异常,执行方法之前的异常是无法捕捉的(例:404)
例1:

@ControllerAdvice  
public class ControllerAdviceTest {  
    @ModelAttribute  
    public User newUser() {  
        System.out.println("============应用到所有@RequestMapping注解方法,在其执行之前把返回值放入Model");  
        return new User();  
    }  
    @InitBinder  
    public void initBinder(WebDataBinder binder) {  
        System.out.println("============应用到所有@RequestMapping注解方法,在其执行之前初始化数据绑定器");  
    }  
    @ExceptionHandler(UnauthenticatedException.class)  
    @ResponseStatus(HttpStatus.UNAUTHORIZED)  
    public String processUnauthenticatedException(NativeWebRequest request, UnauthenticatedException e) {  
        System.out.println("===========应用到所有@RequestMapping注解的方法,在其抛出UnauthenticatedException异常时执行");  
        return "viewName"; //返回一个逻辑视图名  
    }  
}  

例2:

// 可以指定类型,本例就是加了RestController注解的类
@ControllerAdvice(annotations = RestController.class)
public class RestResponseEntityExceptionHandler extends ResponseEntityExceptionHandler {
    private final Logger logger = LoggerFactory.getLogger(this.getClass());
   // 方法1:value是能够被捕获的异常类型,不同的异常调用不同的处理方法
    @ExceptionHandler(value = {
            FatalException.class,
            NonFatalException.class
    })
    protected ResponseEntity<Object> handleExceptions(Exception ex) {
        logger.error("Rest Exception!", ex);
        if (ex instanceof FatalException) {
            return handleFatalException((FatalException) ex);
        } else if (ex instanceof NonFatalException) {
            return handleNonFatalException((NonFatalException) ex);
        } else {
            return new ResponseEntity<Object>(ReturnJson.systemError(), new HttpHeaders(), HttpStatus.INTERNAL_SERVER_ERROR);
        }
    }
    // 处理非致命异常的方法
    private ResponseEntity<Object> handleNonFatalException(NonFatalException ex) {
        return new ResponseEntity<Object>(ReturnJson.json(ex.getCode(), ex.getMessage(), null), new HttpHeaders(), HttpStatus.OK);
    }
    // 处理致命异常的方法
    private ResponseEntity<Object> handleFatalException(FatalException ex) {
        return new ResponseEntity<Object>(ReturnJson.json(ex.getCode(), ex.getMessage(), null), new HttpHeaders(), HttpStatus.INTERNAL_SERVER_ERROR);
    }
    // 方法2:其他异常处理方法
    @ExceptionHandler
    protected ResponseEntity<Object> handleOtherExceptions(Exception ex) {
        logger.error("Rest Exception!", ex);
        return new ResponseEntity<Object>(ReturnJson.systemError(), new HttpHeaders(), HttpStatus.INTERNAL_SERVER_ERROR);
    }
    // 方法3:父类的方法无法满足我们的需要,就要重写父类的方法
    @Override
    protected ResponseEntity<Object> handleMethodArgumentNotValid(MethodArgumentNotValidException ex, HttpHeaders headers, HttpStatus status, WebRequest request) {
        logger.error("请求参数错误", ex);
        return new ResponseEntity<Object>(ReturnJson.fail(), new HttpHeaders(), HttpStatus.OK);
    }
}

ResponseEntityExceptionHandler代码片段

public abstract class ResponseEntityExceptionHandler {
    // 捕获的异常类型,调用不同的方法处理
    @ExceptionHandler(value = {
            NoSuchRequestHandlingMethodException.class,
            HttpRequestMethodNotSupportedException.class,
            HttpMediaTypeNotSupportedException.class,
            HttpMediaTypeNotAcceptableException.class,
            MissingServletRequestParameterException.class,
            ServletRequestBindingException.class,
            ConversionNotSupportedException.class,
            TypeMismatchException.class,
            HttpMessageNotReadableException.class,
            HttpMessageNotWritableException.class,
            MethodArgumentNotValidException.class,
            MissingServletRequestPartException.class,
            BindException.class,
            NoHandlerFoundException.class
    })
    public final ResponseEntity<Object> handleException(Exception ex, WebRequest request) {
        HttpHeaders headers = new HttpHeaders();
        if (ex instanceof NoSuchRequestHandlingMethodException) {
            HttpStatus status = HttpStatus.NOT_FOUND;
            return handleNoSuchRequestHandlingMethod((NoSuchRequestHandlingMethodException) ex, headers, status, request);
        }
        。
        。
        。
        else {
            logger.warn("Unknown exception type: " + ex.getClass().getName());
            HttpStatus status = HttpStatus.INTERNAL_SERVER_ERROR;
            return handleExceptionInternal(ex, null, headers, status, request);
        }
    }
    // 异常处理的方法
    protected ResponseEntity<Object> handleNoSuchRequestHandlingMethod(NoSuchRequestHandlingMethodException ex, HttpHeaders headers, HttpStatus status, WebRequest request) {
        pageNotFoundLogger.warn(ex.getMessage());
        return handleExceptionInternal(ex, null, headers, status, request);
    }
    protected ResponseEntity<Object> handleExceptionInternal(Exception ex, Object body, HttpHeaders headers, HttpStatus status, WebRequest request) {
        if (HttpStatus.INTERNAL_SERVER_ERROR.equals(status)) {
            request.setAttribute("javax.servlet.error.exception", ex, WebRequest.SCOPE_REQUEST);
        }
        return new ResponseEntity<Object>(body, headers, status);
    }
}

对于使用模板渲染HTML的应用

1.在执行@RequestMapping之后遇到的异常

@Component
public class ExceptionHandler implements HandlerExceptionResolver {
    private static Logger log = LoggerFactory.getLogger(ExceptionHandler.class);

    @Override
    public ModelAndView resolveException(HttpServletRequest req, HttpServletResponse resp, Object handler, Exception ex) {
    // 异常处理逻辑 goes here
        log.info("got exception: {}", ex.getClass());
        return new ModelAndView("pc/500");
    }
}

异常的处理逻辑:
DispatcherServlet中如果产生了异常,则接下来会在processDispatchResult()方法中查询当前容器中是否有HandlerExceptionResolver接口的实现类,如果有则调用它的resolveException()方法,得到返回的View,如果没有则使用框架默认的异常处理类。

2.在执行@RequestMapping之前遇到的异常

这个是spring-boot才有的
我们要写一个@Controller,并实现ErrorController接口:

@Controller  
public class MainsiteErrorController implements ErrorController {
 private static final String ERROR_PATH = "/error"; 
 @RequestMapping(value=ERROR_PATH)  
    public String handleError(){  
        return "pages/404";  
    }  
 @Override  
 public String getErrorPath() {
  return ERROR_PATH;  
 }
}  
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring框架中,可以通过全局异常处理机制来统一处理应用程序中的异常情况,避免异常信息直接暴露给用户,提高用户体验和系统的健壮性。以下是一个简单的示例来说明如何实现全局异常处理。 首先,创建一个自定义的全局异常处理器类,该类需要实现`HandlerExceptionResolver`接口。可以通过继承`AbstractHandlerExceptionResolver`类来简化实现。 ```java import org.springframework.web.servlet.HandlerExceptionResolver; import org.springframework.web.servlet.ModelAndView; public class GlobalExceptionHandler implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { // 处理异常逻辑 ModelAndView modelAndView = new ModelAndView(); modelAndView.setViewName("error"); // 设置错误页面 modelAndView.addObject("errorMsg", ex.getMessage()); // 将异常信息传递给错误页面 return modelAndView; } } ``` 接下来,需要在Spring配置文件中注册该全局异常处理器: ```xml <bean id="globalExceptionHandler" class="com.example.GlobalExceptionHandler" /> ``` 最后,可以在控制器中抛出异常,在全局异常处理器中进行统一处理: ```java @Controller public class UserController { @RequestMapping("/user/{id}") public String getUser(@PathVariable("id") int id) { // 模拟抛出异常 if (id <= 0) { throw new IllegalArgumentException("Invalid user ID"); } // ... } } ``` 当控制器方法中抛出异常时,全局异常处理器会捕获该异常并执行相应的处理逻辑。在上述示例中,异常会被处理并渲染到名为"error"的错误页面中。 这样,无论在哪个控制器方法中发生异常,都可以通过全局异常处理器进行统一的异常处理

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值