@ControllerAdvice,是spring3.2提供的新注解,从名字上可以看出大体意思是控制器增强。让我们先看看@ControllerAdvice的实现:
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface ControllerAdvice {
}
没什么特别之处,该注解使用@Component注解,这样的话当我们使用
/**
* Indicates the annotated class assists a "Controller".
*
* Serves as a specialization of {@link Component @Component}, allowing for
* implementation classes to be autodetected through classpath scanning.
*
*
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注解的方法。非常简单,不过只有当使用@ExceptionHandler最有用,另外两个用处不大。
接下来看段代码:
@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"; //返回一个逻辑视图名
}
}
以下为一个实例
package com.zjr.common.web;
//import ...
@ControllerAdvice
public class WebExceptionHandler {
Logger logger = LoggerFactory.getLogger(this.getClass());
@ExceptionHandler({ ServletException.class })
public void http404(HttpServletRequest request, HttpServletResponse response, ServletException e)
throws IOException {
logger.error("HttpRequest error:{}", request.getRequestURL());
logger.error("HttpRequest error stack :" ,e );
outputMessage(response, ErrorType.HTTP404_ERROR.getErrCode(), ErrorType.HTTP404_ERROR.getErrMsg());
}
@ExceptionHandler(BizException.class)
public void handleBizException(HttpServletRequest request, HttpServletResponse response, BizException e)
throws IOException {
logger.error("Biz error occurs during :{}",request.getRequestURI());
logger.error("Biz error stack :",e);
outputMessage(response, e.getErrCode(), e.getMessage());
}
@ExceptionHandler({ Error.class, Exception.class, Throwable.class })
public void exception(HttpServletRequest request, HttpServletResponse response, Throwable e) throws IOException {
logger.error("Unknown error occurs during :{}",request.getRequestURI());
logger.error("Unknown system error stack:", e);
outputMessage(response, ErrorType.UNKNOWN_ERROR.getErrCode(),ErrorType.UNKNOWN_ERROR.getErrMsg());
}
private void outputMessage(HttpServletResponse response, long errCode, String errMsg) throws IOException {
BasicResult result = BasicResult.createFailResult(errCode, errMsg);
String json = new JsonMapper().toJson(result);
response.setCharacterEncoding("UTF-8");
response.setContentType("text/json");
ServletOutputStream os = response.getOutputStream();
os.write(json.getBytes("utf-8"));
}
}