前言
把文章在简书同步了之后,有几篇文章被收入专题,还有一篇文章被收入首页,尽管不是什么了不起的事,但是这样也让自己挺开心的,慕课网约我录了一次视频课程,没有通过,让修改下录第二次,想想还是算了,目前的自己水平还是有限,另一方面时间也比较有限,在博客这方面能够持续下去就已经是件很难的事情,其他的以后再说看缘分吧。接下来继续技术的分析。
前两篇主要记录了AOP所用的核心设计模式-代理模式,包含动态代理和静态代理,以及JDK和CGLIB的两种实现方式,接下来开始重点分析Spring-AOP源码相关操作,当然开始也是从概念理解。这个不太好明白,但是第一遍阅读也主要是为了对于框架有个大局观,不需要过度的拘泥于细节。AOP这段计划在9月中之前结束,而下半月主要是SpringMVC相关的源码研究,可能也会结合一点事务和mybatis源码的研究。
参考了很多程序员DD的东西
Spring-Aop相关概念如下:
我把这个图片放在网络上了,欢迎大家前往下载,建议大家先从地址一下载,地址二要耗流量
下载地址
备用地址
相关的概念描述在脑图中有详细解释,当然方便理解,这里我也会记录下来。
- Target(目标对象):需要被代理增强的对象
- Proxy(代理对象):目标对象被AOP 织入 增强/通知后,产生的对象.
- Joinpoint(连接点):指那些被拦截到的点.在Spring中,这些点指方法(因为Spring只支持方法类型的连接点).
- Pointcut(切入点):指需要(配置)被增强的Joinpoint.
- Advice(通知/增强):指拦截到Joinpoint后要做的操作.通知分为前置通知/后置通知/异常通知/最终通知/环绕通知等.
- Aspect(切面):切入点和通知的结合。
- Weaving(织入):指把增强/通知应用到目标对象来创建代理对象的过程(Spring采用动态代理织入,AspectJ采用编译期织入和类装载期织入).
- Introduction(引入增强):一种特殊通知,在不修改类代码的前提下,可以在运行期为类动态地添加一些Method/Field(不常用).
项目部署
我们先在springboot中将aop整合进来
在pom.xml引入pom依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
starter中默认添加了@EnableAspectJAutoProxy
设计一个简单的controller入门
@RestController
public class HomeController {
private Logger logger = Logger.getLogger(getClass());
@GetMapping("/index")
public String index(@RequestParam String name){
logger.info("-----------{name}:{}"+name);
return "【welcome to aop】:" +name;
}
}
实现Web层的日志切面
- 使用@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.sunliangliang.springsource.controller..*(..))")
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.sunliangliang.springsource.controller.
包下面的所有函数,然后通过@Before
实现,对请求内容的日志记录(本文只是说明过程,可以根据需要调整内容),最后通过@AfterReturning
记录请求返回的对象。
运行程序
输出如下日志:
2017-09-01 18:20:34.953 INFO 14696 --- [nio-8888-exec-3] c.s.springsource.aop.WebLogAspect : URL : http://localhost:8888/index
2017-09-01 18:20:34.953 INFO 14696 --- [nio-8888-exec-3] c.s.springsource.aop.WebLogAspect : HTTP_METHOD : GET
2017-09-01 18:20:34.953 INFO 14696 --- [nio-8888-exec-3] c.s.springsource.aop.WebLogAspect : IP : 0:0:0:0:0:0:0:1
2017-09-01 18:20:34.953 INFO 14696 --- [nio-8888-exec-3] c.s.springsource.aop.WebLogAspect : CLASS_METHOD : com.sunliangliang.springsource.controller.HomeController.index
2017-09-01 18:20:34.953 INFO 14696 --- [nio-8888-exec-3] c.s.springsource.aop.WebLogAspect : ARGS : [liangliang]
2017-09-01 18:20:34.953 INFO 14696 --- [nio-8888-exec-3] c.s.s.controller.HomeController : -----------{name}:{}liangliang
2017-09-01 18:20:34.953 INFO 14696 --- [nio-8888-exec-3] c.s.springsource.aop.WebLogAspect : RESPONSE : 【welcome to aop】:liangliang
优化
由于通过AOP实现,程序得到了很好的解耦,但是也会带来一些问题,比如:我们可能会对Web层做多个切面,校验用户,校验头信息等等,这个时候经常会碰到切面的处理顺序问题。
所以要定义每个切面的优先级,我们需要@Order(i)注解来标识切面的优先级。i值越小,优先级越高。假设我们还有一个切面是CheckNameAspect用来校验name必须为didi,我们为其设置@Order(10),而上文中WebLogAspect设置为@Order(5),所以WebLogAspect有更高的优先级,这个时候执行顺序是这样的:
在@Before中优先执行@Order(5)的内容,再执行@Order(10)的内容
在@After和@AfterReturning中优先执行@Order(10)的内容,再执行@Order(5)的内容
所以我们可以这样子总结:在切入点前的操作,按order的值由小到大执行
在切入点后的操作,按order的值由大到小执行
代码完整示例