浅谈Spring AOP 面向切面编程 最通俗易懂的画图理解AOP、AOP通知执行顺序~

简介

我们都知道,Spring 框架作为后端主流框架之一,最有特点的三部分就是IOC控制反转、依赖注入、以及AOP切面。当然AOP作为一个Spring

的重要组成模块,当然IOC是不依赖于Spring框架的,这就说明你有权选择是否要用AOP来完成一些业务。

AOP面向切面编程,通过另一种思考的方式,来弥补面向对象编程OOP当中的不足,OOP当中最重要的单元是类,所以万物皆对象,万物皆是

对象类。而在AOP的模块单元中,最基础的单元是切面,切面对切点进行模块化的管理。

最后再提一句:Spring当中的AOP是利用Java的代理模式实现的

AOP概念

让我们从一些基础的术语开始了解面向切面编程AOP,术语不是特别的直观,最好的方式就是通过文本理解+图像理解+代码实例理解

这样对于我们来说才是真正意义上的理解。

  • 切面:(Aspect) 一个关注点的模块化,就比较笼统的一个概念,关注点可能横切多个对象。若不理解请往后看图片理解,对应的注解有@Aspect。
  • 连接点:(Joinpoint) 在程序执行过程中某个特定的点,一个连接点总是代表一个方法的执行。
  • 通知:(Advice) 通知表示在一个连接点执行的具体的动作,比如After Before 表明通知的具体动作
  • 切入点:(Pointcut)通过一个表达式去表明我所定义的通知在那个地点具体执行。
  • 前置通知:(Before advice)表明在连接点执行之前执行的动作。
  • 后置通知:(After returning advice)在某个连接点完成后的通知,比如一个方法没有抛出任何异常,正常返回。
  • 环绕通知:(Around Advice) 环绕可以看作是包含前置通知和后置通知的一个通知,先了解,后面具体理解。
  • 异常通知:(After throwing advice) 在方法异常推出时候执行的通知。
  • 最终通知:(After advice) 在连接点退出时候执行的通知。不论是正常退出还是异常退出。

说了这么多,都感觉迷迷糊糊的,我们首先来看一个例子吧,通过这个例子来理解AOP切面,通过例子在具体说明

Springboot AOP

版本信息:Springboot 2.1.6

添加依赖 Web AOP

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>

写一个最简单的控制器

@RestController
@RequestMapping("/")
public class AuthController {

    private Logger logger = LoggerFactory.getLogger(AuthController.class);

    @GetMapping("login")
    public String login(String user,String pass) {

        logger.info("login--->user",user);
        logger.info("login--->pass",pass);

        return "success";
    }

}

这样的控制器在SpringBoot的Web开发当中是很常见的,收到前端的请求,将参数进行校验,我们这里为了简单,不作校验,只是打印出来,为了配合我们后面的AOP的理解而

做的一个最简单的控制器,测试请求以下路径,控制台打印以下内容

http://localhost:8080/login?user=root&pass=root
2019-11-09 09:59:00.585  INFO 11100 --- [nio-8080-exec-1] c.e.demo.controller.AuthController       : login--->user
2019-11-09 09:59:00.588  INFO 11100 --- [nio-8080-exec-1] c.e.demo.controller.AuthController       : login--->pass

定义切面类

1、定义一个切面类,加入@Aspect注解和@Component注解

@Aspect
@Component
public class WebLogAsp {}

@Aspect 注解将找个类定义为一个切面对象,通过@Component注解将这个类对象注入到IOC容器,交给Spring来进行管理。

2、定义一个切入点 通过@Pointcut

    @Pointcut("execution(public * com.example.demo.controller.*.*(..))")
    public void controllerLog(){}

@Pointcut这个注解主要用来定义切入点,通过表达式的方式,来告诉Spring,我这个切点要切到什么位置,常用的就是execution去匹配连接点。

主要来说一下execution 匹配表达是的表达方法,我们按照以下的例子来说明:

@Pointcut("execution(public * com.example.demo.controller.*.*(..))")

语法:execution( [方法修饰符(可选)]__返回类型__类路径__方法名__(参数)__[异常模式(可选)] )

这里我用下划线来代替空格,比较直观的可以看出,我们这个例子里面,我将这个切点切入到com.example.demo.controller包下所有类的所有方法上面。

*就是通配符,(..)代表任意多个参数,也就说明我切入到的方法它的参数我是不限定的,可以有任意个参数。

3、定义通知项,@Before定义一个前置通知

@Before("controllerLog()")
public void beforeAdvice(JoinPoint joinPoint){}

@Before注解传入字符串方法名,也就是我们上面定义的切入点的方法名,告诉它这个通知是在切入点 controllerLog()上面执行的通知,它会在切入点方法执行之前率先执行

这个方法传入了一个JoinPoint 对象,也就是我们所说的连接点对象,连接点可以理解为切入点方法执行时候所产生的一个对象。

这里有几个具体的方法很重要,需要说明一下:

  • Object[] getArgs();获取切入点方法执行时候传入的参数。
  • Object getTarget();获取的是切入方法所在的类对象,简单理解例子里面的切入点是login()方法,所以返回的对象就是AuthController对象
  • Object getThis();返回AOP框架为目标对象生成的代理对象。

这里将详细代码举例,通过打印的方式进行输出,通过SpringMvc的RequestContextHolder获取线程的请求对象,这里我们可以拿到当前请求

的一些具体参数,比如访问人的IP信息,它所请求的URL以及请求所带的参数等待。

具体请参考:https://www.runoob.com/servlet/servlet-client-request.html

    @Before("controllerLog()")
    public void beforeAdvice(JoinPoint joinPoint){

        logger.info("前置通知开始--->");
        /*获取当前请求的HttpServletRequest*/
        RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
        HttpServletRequest request = ((ServletRequestAttributes)requestAttributes).getRequest();

        logger.info("URL-->"+request.getRequestURL().toString());
        logger.info("IP-->"+request.getRemoteAddr());
        logger.info("HTTP_Method-->"+request.getMethod());
        logger.info("Request_args-->"+ Arrays.toString(joinPoint.getArgs()));

        logger.info("前置通知结束--->");
    }

4、环绕通知方法 @Around

首先来看一部分代码,这里传递了一个 ProceedingJoinPoint 对象,它是JoinPoint 切点对象的一个子类,也可以为它是切点所在的方法执行时候所产生的一个对象

这里最重要的一个方法当属于 proceed() ;执行proceed()就表示执行切点所在的方法执行,它会返回一个Object对象,也就是目标方法所返回的对象。

    @Around("controllerLog()")
    public Object aroundAdvice(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        logger.info("环绕通知开始-->");
        Object result = null;

        /*proceed()方法表示连接点方法执行 result 为连接点方法的返回值*/
        result = proceedingJoinPoint.proceed();
        logger.info("环绕通知返回值-->" + result);

        logger.info("环绕通知结束-->");
        return result;
    }

踩坑:环绕方法一定要将result对象返回出去,若定义为void 无返回值的话,将在后面的@AfterReturning 后置通知中无法取到值!!!!

5、后置通知 @AfterReturning 连接点正常执行完毕后的通知

    @AfterReturning(value = "controllerLog()",returning = "obj")
    public void afterReturning(JoinPoint joinPoint,Object obj){
        logger.info("后置通知正常返回-->"+obj);
    }

这里传递了两个参数,第一个参数还是指定切点,第二个参数指定的是返回值,这个返回值就是连接点所返回的参数值,需要在环绕通知里面将其返回出来才可以

取到值,不然返回的就是Null

6、异常通知 @AfterThrowing

    @AfterThrowing(value = "controllerLog()", throwing = "ex")
    public void afterThrowing(JoinPoint joinPoint,Exception ex){
        logger.info("异常通知-->"+ex.getMessage());
    }

这里和上面的后置通知差不多,不过第二个参数是一个异常类型对象,可以取出发生异常时候的异常信息。

7、最终通知 @After

这个和@Before一样,不再细说!

正常返回测试

这里可以很明确的看到,一个正常的返回路径,相信我不说大家都可以看的清楚,画图最清楚的,接收到请求后,首先工作的是环绕通知,

环绕通知里面执行proceed()方法后,才会进入连接点方法执行,Before是在连接点方法执行之前所执行的,而后执行连接点方法内容,

正常执行完后,不发生异常情况,环绕通知结束,就会执行最终通知After After执行完毕后,才会执行后置通知AfterReturning

异常情况测试返回

我们在控制器里面模拟进行一个异常的抛出,看一下执行的顺序

        logger.info("将要抛出异常");
        int a = 1/0;

可以很明明显的看到,我们在控制器里面通过一个除0的操作,进行抛出一个异常,这里环绕通知是没有执行完毕的,因为抛出异常,停止了运行

从接收到请求开始,进入环绕通知,然后环绕通知里面调用了proceed()方法,其实就是让连接点的方法开始运行,这时候前置通知首先跑起来,可以看到

前置通知是完完全全走完的。前置通知完毕后,下来是连接点的方法运行起来了,这里因为抛出了异常,没有进行捕获,最终通知还是一样正常执行,不过最后执行的是异常的通知,

而不是像上面一样。这里不同的地方还是要多进行理解。

画图理解

1、理解切面、切入点、连接点、通知、目标对象

2、理解通知点正常与异常执行顺序

小结

通过对SpringAOP框架的研究,以及画图的理解,能够更深刻的理解里面运行时候的内层含义,以及AOP的一些应用,比如系统里面的日志系统,通过学习AOP

完完全全可以很轻松的通过切面,将需要记录的日志信息存到数据库, 节约大量时间。

参考:

https://docs.spring.io/spring/docs/3.0.x/spring-framework-reference/html/aop-api.html

http://shouce.jb51.net/spring/aop.html

https://www.cnblogs.com/wangshen31/p/9379197.html

代码:

https://gitee.com/mrc1999/Springboot-aop

原文地址:https://www.cnblogs.com/ChromeT/p/11823735.html

时间: 2024-12-20 22:45:36

浅谈Spring AOP 面向切面编程 最通俗易懂的画图理解AOP、AOP通知执行顺序~的相关文章

Spring之面向切面编程AOP(二)

简介 当积累的知识点到一定量的时候,学新知识就变得容易多了.希望再接下来的学习顺利进行下去.今天知识也是挺简单的,主要就是AOP面向切面编程.其中牵涉到了JDKProxy和CGLIB两个代理类,如何使用好,加以深刻理解.学起Spring切面编程也就简单多了 代理模式 1. 代理模式介绍 代理模式的英文叫做Proxy或Surrogate,中文都可译为"代理",所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动.在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对

sprint.net(2) AOP面向切面编程,spring.net的环绕通知;Spring.net的AOP通知的四种类型

AOP 有点类似于我们MVC里面的Filter过滤器,例如在MVC里面,如果给一个Action上打一个标签,就可以在这个Action执行之前或者之后,额外的执行一个方法,这个就相当于是面向切面编程. 无侵入式的. (也就是在不改变原来的代码的情况下,来跳转到一个其他的方法,执行完毕后回到主方法..),但是spring.net的AOP更牛叉,只需要在xml里面配置,就可以了,不需要在方法上面打特性的标签,也不需要继承什么类(例如MVC的过滤器是继承了ActionFilterAttribute) 主

深入探索spring技术内幕(七): 配置Spring AOP面向切面编程

一. AOP一些概念 Aspect( 切面 ): 指横切性关注点的抽象即为切面, 它与类相似, 只是两者的关注点不一样, 类是对物体特征的抽象, 而切面横切性关注点的抽象. joinpoint( 连接点 ): 指那些被拦截到的点. 在spring中, 这些点指的是方法, 因为spring只支持方法类型的连接点, 实际上joinpoint还可以是field或类构造器) Pointcut( 切入点 ): 指我们要对那些joinpoint进行拦截的定义. Advice( 通知 ): 指拦截到joinp

spring入门-AOP 面向切面编程

AOP 面向切面编程 在日常开发中最后将业务逻辑定义在一个专门的service包下,而实现定义在service包下的impl包中, 服务接口以IXXXService形式,而服务实现就是XXXService,这就是规约设计 步骤: 1.E:\Users\WorkSpaces\KmwalletApp\spring\aop\test\HelloWorldService.java 继承(implements)xx接口 2.E:\Users\WorkSpaces\KmwalletApp\spring\ao

spring框架学习(三)——AOP( 面向切面编程)

AOP 即 Aspect Oriented Program 面向切面编程 首先,在面向切面编程的思想里面,把功能分为核心业务功能,和周边功能. 所谓的核心业务,比如登陆,增加数据,删除数据都叫核心业务 所谓的周边功能,比如性能统计,日志,事务管理等等 周边功能在Spring的面向切面编程AOP思想里,即被定义为切面 在面向切面编程AOP的思想里面,核心业务功能和切面功能分别独立进行开发 然后把切面功能和核心业务功能 "编织" 在一起,这就叫AOP 原理图 1. 功能分两大类,辅助功能和

Spring:AOP, 面向切面编程

AOP概述 什么是AOP, 面向切面编程 AOP为Aspect Oriented Programming的缩写, 意为:面向切面编程, 通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术. AOP是OOP的延续, 是函数式编程的一种衍生范型. 利用AOP可以对业务逻辑的各个部分进行隔离, 从而使得业务逻辑各部分之间的耦合度降低, 提高程序的可重用性, 同时提高了开发的效率. - 传统开发模型: 纵向的编程.  面向切面编程: 纵横配合的编程. AOP的作用及优势 作用: 在程序运行期

Javascript aop(面向切面编程)之around(环绕)

Aop又叫面向切面编程,其中“通知”是切面的具体实现,分为before(前置通知).after(后置通知).around(环绕通知),用过spring的同学肯定对它非常熟悉,而在js中,AOP是一个被严重忽视的技术点.但是利用aop可以有效的改善js代码逻辑,比如前端框架dojo和yui3中AOP则被提升至自定义事件的一种内在机制,在源码中随处可见.得益于这种抽象使得dojo的自定义事件异常强大和灵活.dojo中aop的实现在dojo/aspect模块中,主要有三个方法:before.after

Spring-05 -AOP [面向切面编程] -Schema-based 实现aop的步骤

一.AOP [知识点详解] AOP:中文名称面向切面编程 英文名称:(Aspect Oriented Programming) 正常程序执行流程都是纵向执行流程 3.1 又叫面向切面编程,在原有纵向执行流程中添加横切面 3.2 不需要修改原有程序代码 3.2.1 高扩展性 3.2.2 原有功能相当于释放了部分逻辑.让职责更加明确. 面向切面编程是什么? 4.1 在程序原有纵向执行流程中,针对某一个或某一些方法添加通知,形成横切面过程就叫做面向切面编程. 常用概念 5.1 原有功能: 切点, po

Spring IOP 面向切面编程

Spring IOP  面向切面编程 AOP操作术语 Joinpoint(连接点):所谓连接点是指那些被拦截到的点.在spring中,这些点指的是方法,因为spring只支持方法类型的连接点.(类里面可以增强的方法.这些方法称为连接点) Pointcut(切入点):所谓切入点是指我们要对哪些Joinpoint进行拦截的定义. (在类里面可以有很多的方法被增强,比如实际操作中,只是增强了类里面add方法和update方法,实现增强的方法称为切入点) Advice(通知/增强):所谓通知是指拦截到J