• Spring 学习(二)AOP


    一、什么是AOP

    Aspect Oriented Programming,即面向切面编程。对一个大型项目的代码而言,整个系统要求关注安全检查、日志、事务等功能,这些功能实际上“横跨”多个业务方法。在一般的OOP编程里,需要在每一个业务方法内添加相关非业务方法的调用,这实际上是冗余的。如果能够类似IoC一样,这样的安全检查,日志,事务功能单独提取放到外面,核心业务方法不需要关注,就能降低代码耦合度。如果我们以AOP的视角来编写上述业务,可以依次实现:

    • 核心逻辑Service
    • 切面逻辑,即:
      • 权限检查的Aspect;
      • 日志的Aspect;
      • 事务的Aspect。
        然后,以某种方式,让框架来把上述3个Aspect以Proxy的方式“织入”到Service中,这样一来,就不必编写复杂而冗长的Proxy模式

    AOP对于解决特定问题,例如事务管理非常有用,这是因为分散在各处的事务代码几乎是完全相同的,并且它们需要的参数(JDBC的Connection)也是固定的。另一些特定问题,如日志,就不那么容易实现,因为日志虽然简单,但打印日志的时候,经常需要捕获局部变量,如果使用AOP实现日志,我们只能输出固定格式的日志,因此,使用AOP时,必须适合特定的场景。


    二、AOP

    常见的AOP术语包括:

    • 切面(Aspect):横跨多个类和方法的模块,定义了一组横切关注点的行为。
    • 连接点(Join Point):程序执行过程中可以插入切面的特定点,例如方法调用、异常抛出等。
    • 通知(Advice):切面在连接点上执行的行为,包括前置通知、后置通知、环绕通知、异常通知和最终通知等。
    • 切点(Pointcut):定义了一组连接点的表达式,用于确定切面在哪些连接点上执行。
    • 引入(Introduction):允许向现有类添加新的方法或属性。
    • 织入(Weaving):将切面应用到目标对象中,以创建新的代理对象或修改现有对象的字节码。

    例如,编写日志loggin切片

    @Aspect
    @Component
    public class LoggingAspect {
        // 在执行UserService的每个方法前执行:
        @Before("execution(public * com.itranswarp.learnjava.service.UserService.*(..))")
        public void doAccessCheck() {
            System.err.println("[Before] do access check...");
        }
    
        // 在执行MailService的每个方法前后执行:
        @Around("execution(public * com.itranswarp.learnjava.service.MailService.*(..))")
        public Object doLogging(ProceedingJoinPoint pjp) throws Throwable {
            System.err.println("[Around] start " + pjp.getSignature());
            Object retVal = pjp.proceed();
            System.err.println("[Around] done " + pjp.getSignature());
            return retVal;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • @Before()里面的字符串是告诉AspectJ应该在何处执行该方法,这里写的意思是:执行UserService的每个public方法前执行doAccessCheck()代码
    • @Around注解,它和@Before不同,@Around可以决定是否执行目标方法
    • @Aspect注解,表示它的@Before标注的方法需要注入到UserService的每个public方法执行前,@Around标注的方法需要注入到MailService的每个public方法执行前后

    我们需要给@Configuration类加上一个@EnableAspectJAutoProxy注解,Spring的IoC容器看到这个注解,就会自动查找带有@Aspect的Bean,然后根据每个方法的@Before、@Around等注解把AOP注入到特定的Bean中。

    @Configuration
    @ComponentScan
    @EnableAspectJAutoProxy
    public class AppConfig {
        ...
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    三、AOP实现原理

    AOP原理实际上就是一个代理类的实现,这个代理是借由Spring容器实现的,例如,将LoggingAspect.doAccessCheck注入UserService的每个public方法中,可以通过如下实现:

    public UserServiceAopProxy extends UserService {
        private UserService target;
        private LoggingAspect aspect;
    
        public UserServiceAopProxy(UserService target, LoggingAspect aspect) {
            this.target = target;
            this.aspect = aspect;
        }
    
        public User login(String email, String password) {
            // 先执行Aspect的代码:
            aspect.doAccessCheck();
            // 再执行UserService的逻辑:
            return target.login(email, password);
        }
    
        public User register(String email, String password, String name) {
            aspect.doAccessCheck();
            return target.register(email, password, name);
        }
    
        ...
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    容器自动为我们创建了注入了Aspect的子类,取代原始的UserService,并把被@Before @Around关键词修饰的方法覆写。


    四、拦截器

    AOP拦截器通常是指用于拦截和处理方法调用的组件或类。这些拦截器可以在方法调用前、后或异常抛出时执行特定的逻辑

    • @Before:该注解用于定义前置拦截器。在目标方法执行前,被注解的方法将被执行。可以用于实现预处理、参数验证和权限检查等功能。
    • @AfterReturning:该注解用于定义后置拦截器。在目标方法成功执行并返回结果后,被注解的方法将被执行。可以用于实现日志记录、结果处理和清理操作等功能。
    • @Around:该注解用于定义环绕拦截器。在目标方法执行前后,被注解的方法将被执行。通过在拦截器方法中调用ProceedingJoinPoint.proceed()来控制目标方法的执行,并可以在适当的时机添加额外的逻辑。
    • @AfterThrowing:
      该注解用于定义异常拦截器。在目标方法抛出异常时,被注解的方法将被执行。可以用于实现异常处理、错误日志记录等功能。

    通过使用这些注解,可以将拦截器逻辑与特定的切点(Pointcut)相结合,实现对核心业务逻辑的拦截和处理。可以使用execution()表达式来定义切点,指定要拦截的方法的执行。这些注解可以与Spring AOP的其他功能和配置一起使用,如切面(Aspect)、通知(Advice)和配置文件等,以实现更复杂的AOP编程。


    五、注解装配

    使用注解装配AOP时,最好需要在被装配的Bean处使用注解标记,被装配的Bean最好自己能清清楚楚地知道自己被安排了。例如,Spring提供的@Transactional。

    我们以一个实际例子演示如何使用注解实现AOP装配。为了监控应用程序的性能,我们定义一个性能监控的注解:

    @Target(METHOD)
    @Retention(RUNTIME)
    public @interface MetricTime {
        String value();
    }
    
    /**************************************/
    @Component
    public class UserService {
        // 监控register()方法性能:
        @MetricTime("register")
        public User register(String email, String password, String name) {
            ...
        }
        ...
    }
    
    /**************************************/
    @Aspect
    @Component
    public class MetricAspect {
        @Around("@annotation(metricTime)")
        public Object metric(ProceedingJoinPoint joinPoint, MetricTime metricTime) throws Throwable {
            String name = metricTime.value();
            long start = System.currentTimeMillis();
            try {
                return joinPoint.proceed();
            } finally {
                long t = System.currentTimeMillis() - start;
                // 写入日志或发送至JMX:
                System.err.println("[Metrics] " + name + ": " + t + "ms");
            }
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34

    这段代码中,一直对@Around("@annotation(metricTime)")存在疑惑,为什么不是MetricTime(类)而是metricTime(参数名),关键在@annotation。

    @annotation 是 Spring AOP 中的一个切点指示器,用于匹配被任意注解标记的方法或类。在切点表达式中使用 @annotation(annotationType),其中 annotationType 是要匹配的注解类型,可以是任何有效的注解类型。这个切点指示器的作用是,在切面中匹配被特定注解标记的方法或类,以便在相应的切面方法中对它们进行处理。

    以下实例中,@Around(“@annotation(com.example.MyAnnotation)”) 表达式表示匹配被 @MyAnnotation 注解标记的方法。当调用这些被标记的方法时,切面中的 myAdvice 方法会被触发

    @Aspect
    @Component
    public class MyAspect {
    
        // 匹配被 @MyAnnotation 标记的方法
        @Around("@annotation(com.example.MyAnnotation)")
        public Object myAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
            // 在方法执行前后执行自定义逻辑
            // ...
            return joinPoint.proceed();
        }
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    但在@Around("@annotation(metricTime)")中,由于切面方法metric要用到被标记注解的实例,而不只是匹配注解类型。那么在 @annotation() 中需要传入该注解的实例。具体来说,如果被切入的方法的参数列表中有一个具有某个特定注解的参数,则可以在切面表达式中使用 @annotation(parameterName) 来匹配并获取该注解的实例。

  • 相关阅读:
    Hello World分析
    扫雷?拿来吧你(递归展开+坐标标记)
    搭建Nacos集群
    [CSS动效][按钮篇] 适用于 FlatUI 的扁平化按钮
    Vue 前端代码风格指南
    上海亚商投顾:沪指高开低走 钠离子电池、储能概念崛起
    能链科技完成重点项目签约
    oracle安装,导出、导入domp文件、解开oracle行级锁
    FFmpeg开发笔记(三十八)APP如何访问SRS推流的RTMP直播地址
    二叉树题目:二叉树剪枝
  • 原文地址:https://blog.csdn.net/good_jiojio/article/details/132895816