自定义异常拦截一般有两种方式:
1.使用@RestControllerAdvice或者@ControllerAdvice注解定义全局异常拦截
2.基于AOP的异常拦截
使用注解方式
这两种注解都在org.springframework.web.bind.annotation包下面
两者区别:
1)注解有@ControllerAdvice的类, 需要在具体方法上同时添加@ExceptionHandler和@ResponseBody注解;
2)注解有@RestControllerAdvice的类,只需要在具体方法上添加@ExceptionHandler注解。
这里主要展示@RestControllerAdvice的使用
package com.framework.web.exception;
import javax.servlet.http.HttpServletRequest;
import org.apache.shiro.authz.AuthorizationException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.HttpRequestMethodNotSupportedException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import org.springframework.web.servlet.ModelAndView;
import com.common.core.domain.AjaxResult;
import com.common.exception.BusinessException;
import com.common.exception.DemoModeException;
import com.common.utils.ServletUtils;
import com.common.utils.security.PermissionUtils;
/**
* 全局异常处理器
*
*/
@RestControllerAdvice
public class GlobalExceptionHandler
{
private static final Logger log = LoggerFactory.getLogger(GlobalExceptionHandler.class);
/**
* 权限校验失败 如果请求为ajax返回json,普通请求跳转页面
*/
@ExceptionHandler(AuthorizationException.class)
public Object handleAuthorizationException(HttpServletRequest request, AuthorizationException e)
{
log.error(e.getMessage(), e);
if (ServletUtils.isAjaxRequest(request))//此处ServletUtils属于自定义,当前不作说明
{
return AjaxResult.error(PermissionUtils.getMsg(e.getMessage()));
}
else
{
ModelAndView modelAndView = new ModelAndView();
modelAndView.setViewName("error/unauth");
return modelAndView;
}
}
/**
* 请求方式不支持
*/
@ExceptionHandler({ HttpRequestMethodNotSupportedException.class })
public AjaxResult handleException(HttpRequestMethodNotSupportedException e)
{
log.error(e.getMessage(), e);
return AjaxResult.error("不支持' " + e.getMethod() + "'请求");
}
/**
* 拦截未知的运行时异常
*/
@ExceptionHandler(RuntimeException.class)
public AjaxResult notFount(RuntimeException e)
{
log.error("运行时异常:", e);
return AjaxResult.error("运行时异常:" + e.getMessage());
}
/**
* 系统异常
*/
@ExceptionHandler(Exception.class)
public AjaxResult handleException(Exception e)
{
log.error(e.getMessage(), e);
return AjaxResult.error("服务器错误,请联系管理员");
}
/**
* 自定义异常,属于自定义异常BusinessException,然后使用此处拦截
*/
@ExceptionHandler(BusinessException.class)
public Object businessException(HttpServletRequest request, BusinessException e)
{
log.error(e.getMessage(), e);
if (ServletUtils.isAjaxRequest(request))
{
return AjaxResult.error(e.getMessage());
}
else
{
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("errorMessage", e.getMessage());
modelAndView.setViewName("error/business");
return modelAndView;
}
}
}
AOP拦截方式
注解
package com.common.annotation;
import com.common.enums.DataOperateType;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* 异常拦截注解
*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ExceptionIntercept {
DataOperateType operateType(); //操作类型 可以自定义
}
枚举类
package com.common.enums;
/**
* 数据操作类型
*
*/
public enum DataOperateType {
/**
* 修改
*/
UPDATE,
/**
* 插入
*/
INSERT,
/**
* 删除
*/
DELETE
}
切面类
1)pointcut/value:这两个属性的作用是一样的,它们都用于指定该切入点对应的切入表达式。一样既可是一个已有的切入点,也可以直接定义切入点表达式。当指定了pointcut属性后,value属性值将会被覆盖。
2)throwing:该属性指定一个形参名,用于表示Advice方法中可定义与此同名的形参,该形参可用于访问目标方法抛出的异常。除此之外,在Advice方法中定义该参数时,指定的类型,会限制方法必须抛出指定类型的异常。此处使用e,参数中也要使用e,名字保持一致
3)Pointcut:其中有以下两种使用方式,举例:
@within(org.springframework.stereotype.Service),拦截带有 @Service 注解的类的所有方法
@annotation(org.springframework.web.bind.annotation.RequestMapping),拦截带有@RquestMapping的注解方法
package com.framework.aspectj;
import com.common.annotation.ExceptionIntercept;
import com.common.enums.DataOperateType;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.*;
import org.aspectj.lang.reflect.MethodSignature;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import java.lang.reflect.Method;
/**
* 异常拦截处理
*/
@Aspect
@Component
public class ExceptionAspect {
private static final Logger logger = LoggerFactory.getLogger(ExceptionAspect.class);
/**
* 配置织入点,ExceptionIntercept是注解,此处只是拦截有该注解的方法
*/
@Pointcut("@annotation(com.common.annotation.ExceptionIntercept)")
public void exceptionPointCut() {
}
@Before("exceptionPointCut()")
public void doBefore(JoinPoint point) {
// System.out.println("ExceptionAspect的前置通知开始");
}
@AfterReturning("exceptionPointCut()")
public void doAfter(JoinPoint point) {
// System.out.println("ExceptionAspect的后置通知开始");
}
/**
* 抛出异常时执行
* @param point
* @param e
*/
@AfterThrowing(pointcut = "exceptionPointCut()", throwing = "e")
public void handleThrowsIng(JoinPoint point, Exception e) {
Signature signature = point.getSignature();
Object[] args = point.getArgs();
MethodSignature methodSignature = (MethodSignature) signature;
Method method = methodSignature.getMethod();
if (method == null) {
return;
}
//获取注解
ExceptionIntercept annotation = method.getAnnotation(ExceptionIntercept.class);
DataOperateType dataOperateType = annotation.operateType();
if (dataOperateType == null) {
return;
}
//获取方法的参数类型数组
Class[] parameterTypes = methodSignature.getParameterTypes();
String name = method.getDeclaringClass().getName();//类名全称
String params = parameterTypes[0].getName();//第一个参数名(可以获取多个)
String dateType = dataOperateType.name();//参数类型
//TODO 进行下一步操作
}
}
用例:
@ExceptionIntercept(operateType = DataOperateType.DELETE)
public int deleteTRenyuanByIds(String ids) {
return tRenyuanMapper.deleteTRenyuanByIds(Convert.toStrArray(ids));
}
注意事项:
1.切面的执行顺序,与过滤器、拦截器的区别
过滤器 -> 通过集成Filter实现. 缺点如下
无法获取请求要访问的类与方法,以及参数. 可以获取原始的http请求与相应
拦截器 -> 基于springmvc提供的拦截器接口,自定义实现.缺点如下
可以获取请求访问的类与方法 , 但是无法获取请求参数的值. 具体可根据dispatcherServlet跟踪源码
切面 -> 基于spring , 通过aspect注解实现
可以获取访问的类 方法 以及参数值. 但是无法获取http原始的请求与相应的对象
请求处理顺序:
过滤器 -> 拦截器 -> 切面
报错处理顺序:
切面 -> controllerAdvice -> 拦截器 -> 过滤器 -> 服务
2. 被拦截的方法如果无特殊情况不能try catch拦截,否则全局异常拦截类无法获取异常。
3.AOP无法彻底拦截异常,仍然会传递到上一级
AOP的AfterThrowing处理虽然可以对目标方法的异常进行处理,但这种处理与直接使用catch捕捉不同,catch捕捉意味着完全处理该异常,如果catch块中没有重新抛出新的异常,则该方法可能正常结束;而AfterThrowing处理虽然处理了该异常,但它不能完全处理异常,该异常依然会传播到上一级调用者,即JVM。
4.AOP无法拦截内部类方法的调用
说明:
在类C中,方法A调用方法B,B方法被AOP拦截,最终A-》B ,B并不会触发AOP
解决:
第一种解决方案:
1.将当前的代理类暴露给线程使用,以下2种自己选一个实现即可。
注解实现方案:springboot:启动类上加注解:@EnableAspectJAutoProxy(exposeProxy=true):
配置实现方案:<aop:aspectj-autoproxy expose-proxy="true" />
2.A中调用B:不要直接用this(因为this是目标对象,自然无法实现代理类的增强方法@before等),
先去尝试获取代理类:UserServiceImpl service = AopContext.currentProxy() != null ? (UserService)AopContext.currentProxy() : this;
第二种方案:
去除AOP切面增强,把切面方法单独封装接口方法,在需要的地方调用,意思取消AOP方式,使用接口方式,
用例:
该注解拦截使用上面的例子:@ExceptionIntercept,本案例采用第一种方案
@Service
public class TRenyuanServiceImpl implements ITRenyuanService {
@Autowired(required = false)
private TRenyuanMapper tRenyuanMapper;
/**
* 新增人员
* @return
*/
@Override
@ExceptionIntercept(operateType = DataOperateType.INSERT)
public int insertTRenyuan(TRenyuan tRenyuan) {
return tRenyuanMapper.insertTRenyuan(tRenyuan);
}
/**
* 内部类调用
* @param tRenyuan
*/
@Override
public void addRenYuan(TRenyuan tRenyuan) {
/** 第一种方式:如果该方法出现异常,插入失败,但异常不会被@ExceptionIntercept注解拦截*/
this.insertTRenyuan(tRenyuan);
/** 第二种方式:可以拦截 */
ITRenyuanService itRenyuanService = (ITRenyuanService) AopContext.currentProxy();
itRenyuanService.insertTRenyuan(tRenyuan);
}
}