Spring 揭秘之Spring AOP 二世

2024-01-10 00:08
文章标签 java spring aop 揭秘 二世

本文主要是介绍Spring 揭秘之Spring AOP 二世,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • Spring AOP 二世
    • @AspectJ形式的Spring AOP
      • @AspectJ形式的Pointcut
        • @AspectJ形式的Pointcut声明方式
        • @AspectJ形式的Pointcut表达式的标记符
        • @AspectJ形式的Pointcut在Spring AOP中的真实面目
      • @AspectJ形式的Advice
        • Before Advice
        • After Throwing Advice
        • After Returning Advice
        • After Advice
        • Around Advice
        • Introduction
      • @AspectJ中的Aspect更多话题
        • 执行顺序问题
        • 实例化模式问题
    • 说明
    • 小结
    • 展望

Spring AOP 二世

@AspectJ形式的Spring AOP

Spring倡导基于POJO的轻量级编程模式,而Spring框架在1.0的时候并没有提供基于POJO的AOP实现。

好在不久之后,Spring就提供了对AspectJ的支持,从而使得我们可以借助一套标准的注解+POJO完成AOP概念的描述;

虽然,Spring AOP提供了对AspectJ的集成,但是实际上仅仅是利用AspectJ的类库进行Pointcut的解析和匹配,底层的实现机制仍旧是Spring AOP最初的架构;

@AspectJ形式的Pointcut

@AspectJ形式的Pointcut声明方式

@AspectJ形式的Pointcut声明,依附在@Aspect所标注的Aspect定义类中;

@AspectJ形式的Pointcut声明包含两部分:Pointcut Expression和Pointcut Signature。

Pointcut Expression的载体为@Pointcut;该注解为方法级别的注解;Pointcut Expression所在的方法被称为Pointcut Signature;

@Pointcut所指定的表达式分为两部分:

  1. Pointcut标记符:表明该Pointcut以什么样的行为匹配表达式;
  2. 表达视频匹配模式:在Pointcut标记符内制定具体的匹配模式;

Pointcut Signature具化为一个方法,除了返回值需要是void以外,没什么特殊的要求;访问控制符方面,public表示该Pointcut Signature可以在其他Aspect定中使用;private表示只能在当前Aspect中使用;

也可以使用Pointcut Signature在另一Pointcut声明中代替Pointcut Expression,以避免重复定义Pointcut表达式;

AspectJ支持通过&&、||、!等逻辑运算符在Pointcut表达式、Pointcut Signature之间进行逻辑运算,结果仍为Pointcut表达式;

@AspectJ形式的Pointcut表达式的标记符

因为Spring AOP支持支方法级别的Joinpoint,所以可使用的标记符有:

  1. execution:匹配就有指定方法签名的Joinpoint;其格式规定如下:

    execution(modifiers-pattern?ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?);

    在execution中,可以使用两种通配符:*和…;前者可以用于任何部分,匹配相邻的多个字符,也就是一个Word;后者可以在两个提防使用,一个是declaring-type-pattern,表明多个层次的类型;一个是param-pattern,表示0到多个参数,参数类型不限;

    com.xiaomo.*.dianZan(),只能匹配到com.xiaomo包下的类中的dianZan()

  2. within:只接受类型声明,匹配该类型下所有的Joinpoint;同样可以使用*和…来进行通配;

  3. this和target:这两个概念比较复杂。在Aspect中,this指代调用方法的对象,target代表被调用的方法所在的对象;同时使用这两个标识符,可以限定方法的调用关系;在Spring AOP中,this指代代理对象而target指代目标对象;从代理模式来看,代理对象通常和目标对象的类型是相同的;当然也有不一致的情况,需要特别注意;

  4. args:帮助我们捕捉拥有指定参数类型、数量的方法级Joinpoint;通过execution指定的参数,是静态匹配的;而通过args指定的参数则属于动态匹配;比如likeThisArticle(Object user);以及args(User);那么只要Object在运行时是User类型,即可命中;但是通过execute(void likeThisArticle(User))则无法命中;

  5. @Within:目标类使用了指定注解,即可命中该类中所有方法;

  6. @Target:目标对象拥有指定的注解,即可命中该类中所有方法;它和@Within很类似;实际上,@Within属于静态匹配;而@Target则是在运行时动态匹配Joinpoint;

  7. @args:检查当前方法传入的参数是否有指定的注解,如果有,则命中该方法;

  8. @annotaion:检查当前方法时候有指定的注解,如果有,则命中该方法;对注解的要求是只能用于方法级别;

@AspectJ形式的Pointcut在Spring AOP中的真实面目

实际上,所有@AspectJ形式声明的Pointcut表达式,在Spring 内部都将通过解析,转化为具体的Pointcut对象——AspectJExpressionPointcut;
在这里插入图片描述

也就是说,通过AspectJExpressionPointcut,实现了AspectJ和Spring AOP的适配;

@AspectJ形式的Advice

@AspectJ形式的Advice,通过在@Aspect注解的类中的方法加上表示Advice类别的注解来定义;可用注解有:

  1. @Before:对应于Before Advice;
  2. @AfterReturning:对应于After Returning Advice;
  3. @AfterThrowing:对应于ThrowsAdvice;
  4. @After:对应于After(Finally)类别的Advice;
  5. @Around:对应于拦截器类型的Advice;
  6. @DeclareParents:用于标注Introduction类型的Advice;通常用于Field而不是Method;

有了Advice和Pointcut,我们需要建立它们之间的连接;我们可以在Advice注解上指定对应的Pointcut,也就说,将该Advice织入到Pointcut命中的地方;可以指定Pointcut表达式,也可以指定Pointcut Signature;

Before Advice

在某些情况下,我们可能需要在Before Advice定义中访问Joinpoint处的方法参数,我们可以通过以下两种方法实现该目的:

  1. 使用JoinPoint;在@AspectJ形式的Aspect中,我们可以将Advice的第一个参数设置为JoinPoint类型,然后借助其getArgs方法,便可以访问到Joinpoint方法的参数值;getThis()可以获得当前的代理对象;getTarget()可以获得当前目标对象等;

  2. 通过args标记符绑定;当args接受的不是具体的对象类型,而是某个参数的名称时,它将把该参数名称所对应的参数值绑定到Advice方法的调用上;如:

    @Before(value="execution(boolean *.execute(String,..))&&args(tarskName)")
    public void setUpResourceBefore(String taskName) throws Throwable
    

    如此,Before Advice就可以通过taskName获取Joinpoint处的taskName参数啦;

实际上,除了Around Advice和Introduction不可以这么用外,其余的Advice均可以按照这种方式解决参数访问问题;值得注意的是,JoinPoint永远处于第一个参数位置;

另外,除了args外,this、target、@within、@target、@annotation等在指定参数名称的时候,就可以把相关内容绑定到对应的Advice的方法参数上;

After Throwing Advice

该类型的Advice有一个throwing属性,通过它,我们可以限定Advice定义方法的参数名,并在方法调用的时候,将相应的异常传递到具体方法参数上;

After Returning Advice

我们可以通过该类型的Advice的returning属性访问到Joinpoint处的返回值;

After Advice

并没有什么需要特别说明的地方;

Around Advice

对于Around Advice所标注的方法来说,其第一个参数必须是ProceedingJoinPoint;因为我们需要调用该类型参数的proceed()方法继续调用链的执行;

Introduction

以上Advice都是通过注解对方法进行标注,但是Introduction完全不同,因为它是对Aspect中的实例变量进行标注;

以AspectJ形式声明Introduction时,我们需要在Aspect中声明一个实例变量;它的类型就是新增功能的接口类型;然后使用@DeclareParents对其进行注解,同时通过defaultImpl属性指定该接口的实现类,通过value指定目标对象;

@AspectJ中的Aspect更多话题

执行顺序问题

执行顺序问题:当多个Advice命中同一Joinpoint的时候,它们的执行顺序如何决定?

如果这些Advice都声明在同一个Aspect中,那么**执行顺序由这些Advice的声明顺序决定;**最先声明的Advice拥有最高的优先权;对于Before Advice来说,最高优先级的方法最先运行;对于AfterReturningAdvice来说,最高优先的方法最后运行;

如果这些Advice分别属于不同的Aspect中,那么,我们就需要Ordered接口啦;否则,执行顺序无法确定;

注意!如果通过Spring的IoC容器注册并使用这些Aspect,让自动代理机制处理这些横切逻辑的织入过程,那么情况就是上述情况啦;如果是通过编程方式使用Aspect,那么Aspect内的Advice执行顺序完全由添加到AspectJProxyFactory的顺序来决定;

实例化模式问题

AspectJ默认使用Singleton模式,同时Spring AOP还支持perthis和pertarget等实例化模式;

我们可以在@Aspect注解里指定Aspect的实例化模式;如:

@Aspect("perthis(exection(boolean *.execute(String,..)))")
public class MultiAdvicesAspect(){@Pointcut("execution(boolean *.execute(String,..))")public void taskExecution(){}
}

perthis将为相应的代理对象实例化各自的Aspect实例;对于pertarget则为匹配的单独的目标对象实例化相应的Aspect实例;

说明

在《Spring揭秘》中,以上内容为该章的第一部分,第二部分讲解基于Schema的AspectJ支持;考虑到目前注解方式为主流的使用方式,我们就总结后面的内容啦;

小结

hhh,强行小结;至此,Spring的第二辆马车——Spring AOP就介绍完毕啦;

回顾我们对Spring的学习路线:IoC Service Provder——Spring IoC容器(包含BeanFactory和ApplicationContext)——AOP基本概念——SPring AOP的实现(底层结构以及对AspectJ的支持);我们会发现,这是一个从一般到特殊的过程

这样的安排是很合理的,一方面它符合人们的认知习惯,在学习掌握相关知识的过程中不会有“迷失感”,让人感到很“踏实”;另一方面它也符合面向对象的继承实现策略——从抽象到具体;所以,该书的章节安排不可谓不科学,不可谓不精彩;

然而,这本经典书籍成书已久,所使用的Spring版本在目前看来就比较旧了(当时,也是最新版呢);这也是为啥我们贴的代码比较少,而理论叙述比较多的原因之一;(博文中的源码均来自Spring5.1,示例代码部分源自书本,一些系博主原创);

展望

Spring框架的核心概念——IoC和AOP我们已经大体上掌握,接下来我们将采用“双线并行”的策略:一方面学习Spring框架的具体应用,以Spring MVC框架为学习样本;另一方面学习、阅读最新的Spring文档;

这篇关于Spring 揭秘之Spring AOP 二世的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/588889

相关文章

Spring Boot整合Redis注解实现增删改查功能(Redis注解使用)

《SpringBoot整合Redis注解实现增删改查功能(Redis注解使用)》文章介绍了如何使用SpringBoot整合Redis注解实现增删改查功能,包括配置、实体类、Repository、Se... 目录配置Redis连接定义实体类创建Repository接口增删改查操作示例插入数据查询数据删除数据更

Java Lettuce 客户端入门到生产的实现步骤

《JavaLettuce客户端入门到生产的实现步骤》本文主要介绍了JavaLettuce客户端入门到生产的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 目录1 安装依赖MavenGradle2 最小化连接示例3 核心特性速览4 生产环境配置建议5 常见问题

Java使用Swing生成一个最大公约数计算器

《Java使用Swing生成一个最大公约数计算器》这篇文章主要为大家详细介绍了Java使用Swing生成一个最大公约数计算器的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下... 目录第一步:利用欧几里得算法计算最大公约数欧几里得算法的证明情形 1:b=0情形 2:b>0完成相关代码第二步:加

Java 的ArrayList集合底层实现与最佳实践

《Java的ArrayList集合底层实现与最佳实践》本文主要介绍了Java的ArrayList集合类的核心概念、底层实现、关键成员变量、初始化机制、容量演变、扩容机制、性能分析、核心方法源码解析、... 目录1. 核心概念与底层实现1.1 ArrayList 的本质1.1.1 底层数据结构JDK 1.7

Java Map排序如何按照值按照键排序

《JavaMap排序如何按照值按照键排序》该文章主要介绍Java中三种Map(HashMap、LinkedHashMap、TreeMap)的默认排序行为及实现按键排序和按值排序的方法,每种方法结合实... 目录一、先理清 3 种 Map 的默认排序行为二、按「键」排序的实现方式1. 方式 1:用 TreeM

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node