实现AOP的切面主要有如下几个要素:html
@Aspect
注解将一个java类定义为切面类@Pointcut
定义一个切入点,能够是一个规则表达式,好比下例中某个package下的全部函数,也能够是一个注解等。@Before
在切入点开始处切入内容@After
在切入点结尾处切入内容@AfterReturning
在切入点return内容以后切入内容(能够用来对处理返回值作一些加工处理)@Around
在切入点先后切入内容,并本身控制什么时候执行切入点自身的内容@AfterThrowing
用来处理当切入内容部分抛出异常以后的处理逻辑
@Aspect @Component public class WebLogAspect { private Logger logger = Logger.getLogger(getClass()); @Pointcut("execution(public * com.didispace.web..*.*(..))") public void webLog(){} @Before("webLog()") public void doBefore(JoinPoint joinPoint) throws Throwable { // 接收到请求,记录请求内容 ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); HttpServletRequest request = attributes.getRequest(); // 记录下请求内容 logger.info("URL : " + request.getRequestURL().toString()); logger.info("HTTP_METHOD : " + request.getMethod()); logger.info("IP : " + request.getRemoteAddr()); logger.info("CLASS_METHOD : " + joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName()); logger.info("ARGS : " + Arrays.toString(joinPoint.getArgs())); } @AfterReturning(returning = "ret", pointcut = "webLog()") public void doAfterReturning(Object ret) throws Throwable { // 处理完请求,返回内容 logger.info("RESPONSE : " + ret); } }
能够看上面的例子,经过@Pointcut
定义的切入点为com.didispace.web
包下的全部函数(对web层全部请求处理作切入点),而后经过@Before
实现,对请求内容的日志记录(本文只是说明过程,能够根据须要调整内容),最后经过@AfterReturning
记录请求返回的对象。java
经过运行程序并访问:http://localhost:8080/hello?name=didi
,能够得到下面的日志输出web
2016-05-19 13:42:13,156 INFO WebLogAspect:41 - URL : http://localhost:8080/hello 2016-05-19 13:42:13,156 INFO WebLogAspect:42 - HTTP_METHOD : http://localhost:8080/hello 2016-05-19 13:42:13,157 INFO WebLogAspect:43 - IP : 0:0:0:0:0:0:0:1 2016-05-19 13:42:13,160 INFO WebLogAspect:44 - CLASS_METHOD : com.didispace.web.HelloController.hello 2016-05-19 13:42:13,160 INFO WebLogAspect:45 - ARGS : [didi] 2016-05-19 13:42:13,170 INFO WebLogAspect:52 - RESPONSE:Hello didi
在WebLogAspect切面中,分别经过doBefore和doAfterReturning两个独立函数实现了切点头部和切点返回后执行的内容,若咱们想统计请求的处理时间,就须要在doBefore处记录时间,并在doAfterReturning处经过当前时间与开始处记录的时间计算获得请求处理的消耗时间。函数
那么咱们是否能够在WebLogAspect切面中定义一个成员变量来给doBefore和doAfterReturning一块儿访问呢?是否会有同步问题呢?优化
的确,直接在这里定义基本类型会有同步问题,因此咱们能够引入ThreadLocal对象,像下面这样进行记录:spa
@Aspect @Component public class WebLogAspect { private Logger logger = Logger.getLogger(getClass()); ThreadLocal<Long> startTime = new ThreadLocal<>(); @Pointcut("execution(public * com.didispace.web..*.*(..))") public void webLog(){} @Before("webLog()") public void doBefore(JoinPoint joinPoint) throws Throwable { startTime.set(System.currentTimeMillis()); // 省略日志记录内容 } @AfterReturning(returning = "ret", pointcut = "webLog()") public void doAfterReturning(Object ret) throws Throwable { // 处理完请求,返回内容 logger.info("RESPONSE : " + ret); logger.info("SPEND TIME : " + (System.currentTimeMillis() - startTime.get())); } }
因为经过AOP实现,程序获得了很好的解耦,可是也会带来一些问题,好比:咱们可能会对Web层作多个切面,校验用户,校验头信息等等,这个时候常常会碰到切面的处理顺序问题。.net
因此,咱们须要定义每一个切面的优先级,咱们须要@Order(i)
注解来标识切面的优先级。i的值越小,优先级越高。假设咱们还有一个切面是CheckNameAspect
用来校验name必须为didi,咱们为其设置@Order(10)
,而上文中WebLogAspect设置为@Order(5)
,因此WebLogAspect有更高的优先级,这个时候执行顺序是这样的:日志
在@Before
中优先执行@Order(5)
的内容,再执行@Order(10)
的内容code
@After
和@AfterReturning
中优先执行@Order(10)
的内容,再执行@Order(5)
的内容因此咱们能够这样子总结:htm