美文网首页spring aopAOP(面向切面编程)Spring Boot
Spring源码-AOP(三)Springboot与Aop

Spring源码-AOP(三)Springboot与Aop

作者: 阿亮私语 | 来源:发表于2017-09-01 18:45 被阅读375次

    前言

    把文章在简书同步了之后,有几篇文章被收入专题,还有一篇文章被收入首页,尽管不是什么了不起的事,但是这样也让自己挺开心的,慕课网约我录了一次视频课程,没有通过,让修改下录第二次,想想还是算了,目前的自己水平还是有限,另一方面时间也比较有限,在博客这方面能够持续下去就已经是件很难的事情,其他的以后再说看缘分吧。接下来继续技术的分析。

    前两篇主要记录了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的值由大到小执行
      代码完整示例

    相关文章

      网友评论

        本文标题:Spring源码-AOP(三)Springboot与Aop

        本文链接:https://www.haomeiwen.com/subject/zzpkjxtx.html