网易首页 > 网易号 > 正文 申请入驻

彻底搞懂了这道必考面试题Spring AOP 是怎么运行的?看完就懂

0
分享至

前言

其实, 接触了这么久的 AOP, 我感觉, AOP 给人难以理解的一个关键点是它的概念比较多, 而且坑爹的是, 这些概念经过了中文翻译后, 变得面目全非, 相同的一个术语, 在不同的翻译下, 含义总有着各种莫名其妙的差别. 鉴于此, 我在本章的开头, 着重为为大家介绍一个 Spring AOP 的各项术语的基本含义. 为了术语传达的准确性, 我在接下来的叙述中, 能使用英文术语的地方, 尽量使用英文。

什么是 AOP

AOP(Aspect-Oriented Programming), 即 面向切面编程 , 它与 OOP( Object-Oriented Programming, 面向对象编程) 相辅相成, 提供了与 OOP 不同的抽象软件结构的视角. 在 OOP 中, 我们以类(class)作为我们的基本单元, 而 AOP 中的基本单元是 Aspect(切面)。

Aspect(切面)

aspect 由 pointcount 和 advice 组成, 它既包含了横切逻辑的定义, 也包括了连接点的定义. Spring AOP就是负责实施切面的框架, 它将切面所定义的横切逻辑织入到切面所指定的连接点中. AOP的工作重心在于如何将增强织入目标对象的连接点上, 这里包含两个工作:

1. 如何通过 pointcut 和 advice 定位到特定的 joinpoint 上。

2. 如何在 advice 中编写切面代码。

可以简单地认为, 使用 @Aspect 注解的类就是切面。

advice(增强)

由 aspect 添加到特定的 join point(即满足 point cut 规则的 join point) 的一段代码. 许多 AOP框架, 包括 Spring AOP, 会将 advice 模拟为一个拦截器(interceptor), 并且在 join point 上维护多个 advice, 进行层层拦截. 例如 HTTP 鉴权的实现, 我们可以为每个使用 RequestMapping 标注的方法织入 advice, 当 HTTP 请求到来时, 首先进入到 advice 代码中, 在这里我们可以分析这个 HTTP 请求是否有相应的权限, 如果有, 则执行 Controller, 如果没有, 则抛出异常. 这里的 advice 就扮演着鉴权拦截器的角色了。

连接点(join point)

a point during the execution of a program, such as the execution of a method or the handling of an exception. In Spring AOP, a join point always represents a method execution.

程序运行中的一些时间点, 例如一个方法的执行, 或者是一个异常的处理.在 Spring AOP 中, join point 总是方法的执行点, 即只有方法连接点。

切点(point cut)

匹配 join point 的谓词(a predicate that matches join points). Advice 是和特定的 point cut 关联的, 并且在 point cut 相匹配的 join point 中执行.在 Spring 中, 所有的方法都可以认为是 joinpoint, 但是我们并不希望在所有的方法上都添加 Advice, 而 pointcut 的作用就是提供一组规则(使用 AspectJ pointcut expression language 来描述) 来匹配joinpoint, 给满足规则的 joinpoint 添加 Advice。

关于join point 和 point cut 的区别

在 Spring AOP 中, 所有的方法执行都是 join point. 而 point cut 是一个描述信息, 它修饰的是 join point, 通过 point cut, 我们就可以确定哪些 join point 可以被织入 Advice. 因此 join point 和 point cut 本质上就是两个不同纬度上的东西.advice 是在 join point 上执行的, 而 point cut 规定了哪些 join point 可以执行哪些 advice。

introduction

为一个类型添加额外的方法或字段. Spring AOP 允许我们为 目标对象 引入新的接口(和对应的实现). 例如我们可以使用 introduction 来为一个 bean 实现 IsModified 接口, 并以此来简化 caching 的实现。

目标对象(Target)

织入 advice 的目标对象. 目标对象也被称为 advised object.因为 Spring AOP 使用运行时代理的方式来实现 aspect, 因此 adviced object 总是一个代理对象(proxied object)注意, adviced object 指的不是原来的类, 而是织入 advice 后所产生的代理类。

AOP proxy

一个类被 AOP 织入 advice, 就会产生一个结果类, 它是融合了原类和增强逻辑的代理类. 在 Spring AOP 中, 一个 AOP 代理是一个 JDK 动态代理对象或 CGLIB 代理对象。

织入(Weaving)

将 aspect 和其他对象连接起来, 并创建 adviced object 的过程. 根据不同的实现技术, AOP织入有三种方式:

  • 编译器织入, 这要求有特殊的Java编译器。
  • 类装载期织入, 这需要有特殊的类装载器。
  • 动态代理织入, 在运行期为目标类添加增强(Advice)生成子类的方式. Spring 采用动态代理织入, 而AspectJ采用编译器织入和类装载期织入。

advice 的类型

  • before advice, 在 join point 前被执行的 advice. 虽然 before advice 是在 join point 前被执行, 但是它并不能够阻止 join point 的执行, 除非发生了异常(即我们在 before advice 代码中, 不能人为地决定是否继续执行 join point 中的代码)。
  • after return advice, 在一个 join point 正常返回后执行的 advice。
  • after throwing advice, 当一个 join point 抛出异常后执行的 advice。
  • after(final) advice, 无论一个 join point 是正常退出还是发生了异常, 都会被执行的 advice。
  • around advice, 在 join point 前和 joint point 退出后都执行的 advice. 这个是最常用的 advice。

关于 AOP Proxy

Spring AOP 默认使用标准的 JDK 动态代理(dynamic proxy)技术来实现 AOP 代理, 通过它, 我们可以为任意的接口实现代理.如果需要为一个类实现代理, 那么可以使用 CGLIB 代理. 当一个业务逻辑对象没有实现接口时, 那么Spring AOP 就默认使用 CGLIB 来作为 AOP 代理了. 即如果我们需要为一个方法织入 advice, 但是这个方法不是一个接口所提供的方法, 则此时 Spring AOP 会使用 CGLIB 来实现动态代理. 鉴于此, Spring AOP 建议基于接口编程, 对接口进行 AOP 而不是类。

彻底理解 aspect, join point, point cut, advice

看完了上面的理论部分知识, 我相信还是会有不少朋友感觉到 AOP 的概念还是很模糊, 对 AOP 中的各种概念理解的还不是很透彻. 其实这很正常, 因为 AOP 中的概念是在是太多了, 我当时也是花了老大劲才梳理清楚的. 下面我以一个简单的例子来比喻一下 AOP 中 aspect, jointpoint, pointcut 与 advice 之间的关系。

让我们来假设一下, 从前有一个叫爪哇的小县城, 在一个月黑风高的晚上, 这个县城中发生了命案. 作案的凶手十分狡猾, 现场没有留下什么有价值的线索. 不过万幸的是, 刚从隔壁回来的老王恰好在这时候无意中发现了凶手行凶的过程, 但是由于天色已晚, 加上凶手蒙着面, 老王并没有看清凶手的面目, 只知道凶手是个男性, 身高约七尺五寸. 爪哇县的县令根据老王的描述, 对守门的士兵下命令说: 凡是发现有身高七尺五寸的男性, 都要抓过来审问. 士兵当然不敢违背县令的命令, 只好把进出城的所有符合条件的人都抓了起来。

来让我们看一下上面的一个小故事和 AOP 到底有什么对应关系. 首先我们知道, 在 Spring AOP 中 join point 指代的是所有方法的执行点, 而 point cut 是一个描述信息, 它修饰的是 join point, 通过 point cut, 我们就可以确定哪些 join point 可以被织入 Advice. 对应到我们在上面举的例子, 我们可以做一个简单的类比, join point 就相当于 爪哇的小县城里的百姓 , point cut 就相当于 老王所做的指控, 即凶手是个男性, 身高约七尺五寸 , 而 advice 则是施加在符合老王所描述的嫌疑人的动作: 抓过来审问 . 为什么可以这样类比呢?

  • aspect: aspect 是 point cut 与 advice 的组合, 因此在这里我们就可以类比: "根据老王的线索, 凡是发现有身高七尺五寸的男性, 都要抓过来审问" 这一整个动作可以被认为是一个 aspect。

或则我们也可以从语法的角度来简单类比一下. 我们在学英语时, 经常会接触什么 定语, 被动句 之类的概念, 那么可以做一个不严谨的类比, 即 joinpoint 可以认为是一个 宾语, 而 pointcut 则可以类比为修饰 joinpoint 的定语, 那么整个 aspect 就可以描述为: 满足 pointcut 规则的 joinpoint 会被添加相应的 advice 操作。

@AspectJ 支持

@AspectJ 是一种使用 Java 注解来实现 AOP 的编码风格. @AspectJ 风格的 AOP 是 AspectJ Project 在 AspectJ 5 中引入的, 并且 Spring 也支持@AspectJ 的 AOP 风格。

使能 @AspectJ 支持

@AspectJ 可以以 XML 的方式或以注解的方式来使能, 并且不论以哪种方式使能@ASpectJ, 我们都必须保证 aspectjweaver.jar 在 classpath 中。

使用 Java Configuration 方式使能@AspectJ

@Configuration

@EnableAspectJAutoProxypublic class AppConfig {}

使用 XML 方式使能@AspectJ

定义 aspect(切面)

当使用注解 @Aspect 标注一个 Bean 后, 那么 Spring 框架会自动收集这些 Bean, 并添加到 Spring AOP 中, 例如:

@Component@Aspectpublicclass MyTest {}

注意, 仅仅使用@Aspect 注解, 并不能将一个 Java 对象转换为 Bean, 因此我们还需要使用类似 @Component 之类的注解.注意, 如果一个 类被@Aspect 标注, 则这个类就不能是其他 aspect 的 advised object 了, 因为使用 @Aspect 后, 这个类就会被排除在 auto-proxying 机制之外。

声明 pointcut

一个 pointcut 的声明由两部分组成:

  • 一个方法签名, 包括方法名和相关参数。
  • 一个 pointcut 表达式, 用来指定哪些方法执行是我们感兴趣的(即因此可以织入 advice)。

在@AspectJ 风格的 AOP 中, 我们使用一个方法来描述 pointcut, 即:

@Pointcut("execution(* com.xys.service.UserService.*(..))") // 切点表达式private void dataAccessOperation() {} // 切点前面

这个方法必须无返回值.这个方法本身就是 pointcut signature, pointcut 表达式使用@Pointcut 注解指定.上面我们简单地定义了一个 pointcut, 这个 pointcut 所描述的是: 匹配所有在包 com.xys.service.UserService 下的所有方法的执行。

切点标志符(designator)

AspectJ5 的切点表达式由标志符(designator)和操作参数组成. 如 "execution( greetTo(..))" 的切点表达式, execution 就是 标志符, 而圆括号里的 greetTo(..) 就是操作参数。

execution

匹配 join point 的执行, 例如 "execution(* hello(..))" 表示匹配所有目标类中的 hello() 方法. 这个是最基本的 pointcut 标志符。

within

匹配特定包下的所有 join point, 例如 within(com.xys.*) 表示 com.xys 包中的所有连接点, 即包中的所有类的所有方法。 而 within(com.xys.service.*Service) 表示在 com.xys.service 包中所有以 Service 结尾的类的所有的连接点。

this 与 target

this 的作用是匹配一个 bean, 这个 bean(Spring AOP proxy) 是一个给定类型的实例(instance of). 而 target 匹配的是一个目标对象(target object, 即需要织入 advice 的原始的类), 此对象是一个给定类型的实例(instance of)。

bean

匹配 bean 名字为指定值的 bean 下的所有方法, 例如:

bean(*Service) // 匹配名字后缀为 Service 的 bean 下的所有方法

bean(myService) // 匹配名字为 myService 的 bean 下的所有方法

args

匹配参数满足要求的的方法. 例如:

@Pointcut("within(com.xys.demo2.*)")public void pointcut2() {}@Before(value = "pointcut2() && args(name)")public void doSomething(String name){ logger.info("---page: {}---", name);}@Servicepublic class NormalService { private Logger logger = LoggerFactory.getLogger(getClass()); public void someMethod() { logger.info("---NormalService: someMethod invoked---"); } public String test(String name) { logger.info("---NormalService: test invoked---"); return "服务一切正常"; }}

当 NormalService.test 执行时, 则 advice doSomething 就会执行, test 方法的参数 name 就会传递到 doSomething 中。

常用例子:

// 匹配只有一个参数 name 的方法@Before(value = "aspectMethod() && args(name)")public void doSomething(String name) {}// 匹配第一个参数为 name 的方法@Before(value = "aspectMethod() && args(name, ..)")public void doSomething(String name) {}// 匹配第二个参数为 name 的方法Before(value = "aspectMethod() && args(*, name, ..)")public void doSomething(String name) {}

@annotation

匹配由指定注解所标注的方法, 例如:

@Pointcut("@annotation(com.xys.demo1.AuthChecker)")public void pointcut() {}

则匹配由注解 AuthChecker 所标注的方法。

常见的切点表达式

匹配方法签名

// 匹配指定包中的所有的方法execution(* com.xys.service.*(..))// 匹配当前包中的指定类的所有方法execution(* UserService.*(..))// 匹配指定包中的所有 public 方法execution(public * com.xys.service.*(..))// 匹配指定包中的所有 public 方法, 并且返回值是 int 类型的方法execution(public int com.xys.service.*(..))// 匹配指定包中的所有 public 方法, 并且第一个参数是 String, 返回值是 int 类型的方法execution(public int com.xys.service.*(String name, ..))匹配类型签名// 匹配指定包中的所有的方法, 但不包括子包within(com.xys.service.*)// 匹配指定包中的所有的方法, 包括子包within(com.xys.service..*)// 匹配当前包中的指定类中的方法within(UserService)// 匹配一个接口的所有实现类中的实现的方法within(UserDao+)匹配 Bean 名字// 匹配以指定名字结尾的 Bean 中的所有方法bean(*Service)切点表达式组合// 匹配以 Service 或 ServiceImpl 结尾的 beanbean(*Service || *ServiceImpl)// 匹配名字以 Service 结尾, 并且在包 com.xys.service 中的 beanbean(*Service) && within(com.xys.service.*)

声明 advice

advice 是和一个 pointcut 表达式关联在一起的, 并且会在匹配的 join point 的方法执行的前/后/周围 运行. pointcut 表达式可以是简单的一个 pointcut 名字的引用, 或者是完整的 pointcut 表达式. 下面我们以几个简单的 advice 为例子, 来看一下一个 advice 是如何声明的。

Before advice

/*** @author xiongyongshun * @version 1.0 * @created 16/9/9 13:13*/@Component@Aspectpublic class BeforeAspectTest { // 定义一个 Pointcut, 使用 切点表达式函数 来描述对哪些 Join point 使用 advise. @Pointcut("execution(* com.xys.service.UserService.*(..))") public void dataAccessOperation() { }}@Component@Aspectpublic class AdviseDefine { // 定义 advise @Before("com.xys.aspect.PointcutDefine.dataAccessOperation()") public void doBeforeAccessCheck(JoinPoint joinPoint) { System.out.println("*****Before advise, method: " + joinPoint.getSignature().toShortString() + " *****"); }}

这里, @Before 引用了一个 pointcut, 即 "com.xys.aspect.PointcutDefine.dataAccessOperation()" 是一个 pointcut 的名字。

如果我们在 advice 在内置 pointcut, 则可以:

@Component@Aspectpublic class AdviseDefine { // 将 pointcut 和 advice 同时定义 @Before("within(com.xys.service..*)") public void doAccessCheck(JoinPoint joinPoint) { System.out.println("*****doAccessCheck, Before advise, method: " + joinPoint.getSignature().toShortString() + " *****"); }}around advicearound advice 比较特别, 它可以在一个方法的之前之前和之后添加不同的操作, 并且甚至可以决定何时, 如何, 是否调用匹配到的方法.@Component@Aspectpublic class AdviseDefine { // 定义 advise @Around("com.xys.aspect.PointcutDefine.dataAccessOperation()") public Object doAroundAccessCheck(ProceedingJoinPoint pjp) throws Throwable { StopWatch stopWatch = new StopWatch(); stopWatch.start(); // 开始 Object retVal = pjp.proceed(); stopWatch.stop(); // 结束 System.out.println("invoke method: " + pjp.getSignature().getName() + ", elapsed time: " + stopWatch.getTotalTimeMillis()); return retVal; }}

around advice 和前面的 before advice 差不多, 只是我们把注解 @Before 改为了 @Around 了。

今天就分享到这里,另外金三银四马上到了,相信很多朋友都准备跳槽面试,所以给大家分享一份Java面试宝典,近千道面试题等你来攻破,每道题目都带有详细的解析答案

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
69岁女港星刚做手术就北上逛街,服装一言难尽,结三次婚却无子女

69岁女港星刚做手术就北上逛街,服装一言难尽,结三次婚却无子女

八卦先生
2024-05-01 23:31:41
这一番中东的和解,为何又在中国达成

这一番中东的和解,为何又在中国达成

新民周刊
2024-05-01 09:38:10
只要捂住,就能蒙混过去?辽宁616名省人大代表523人收钱!

只要捂住,就能蒙混过去?辽宁616名省人大代表523人收钱!

天闻地知
2024-05-02 10:11:34
太阳老板:我们未来8年有5首轮 若想的话下个月就会交易其中的2个

太阳老板:我们未来8年有5首轮 若想的话下个月就会交易其中的2个

直播吧
2024-05-02 10:59:12
事实证明,坚持“丁克”25年的康辉,如今也走上了一条“不归路”

事实证明,坚持“丁克”25年的康辉,如今也走上了一条“不归路”

小兔子爱旅游
2024-04-24 18:02:57
5月1日,杨颖突然冲上了热搜,只因为有网友在云南偶遇了杨颖

5月1日,杨颖突然冲上了热搜,只因为有网友在云南偶遇了杨颖

穷游天下的阿等
2024-05-01 21:24:42
女子在办公室被殴打后续:上衣被扒掉,知情者曝内幕,当小三

女子在办公室被殴打后续:上衣被扒掉,知情者曝内幕,当小三

180°视角
2024-05-01 08:20:34
农业大学教授提议:乡村不能再禁这禁那了!否则会把乡村管死

农业大学教授提议:乡村不能再禁这禁那了!否则会把乡村管死

鹏飞深文
2024-04-29 18:09:30
继续故步自封!日媒:中国汽车越成功,就距离世界越远

继续故步自封!日媒:中国汽车越成功,就距离世界越远

阿珂谈汽车
2024-05-01 13:11:34
江西:37岁大龄剩女吐槽:快绝经了,没人和我结婚,只要38万彩礼

江西:37岁大龄剩女吐槽:快绝经了,没人和我结婚,只要38万彩礼

大有看头
2024-05-02 11:10:03
妈妈晒女儿“袜子破洞”,引来3000条说教,拿不出手的爱就别拿了

妈妈晒女儿“袜子破洞”,引来3000条说教,拿不出手的爱就别拿了

熙熙说教
2024-05-02 10:00:10
那些未曾接种一针疫苗的人,此刻竟满怀得意,甚至自认为独具慧眼

那些未曾接种一针疫苗的人,此刻竟满怀得意,甚至自认为独具慧眼

荷兰豆爱健康
2024-05-02 10:54:20
1993年,中俄列车遭暴徒洗劫强奸140个小时,邓公批示:跨国抓捕

1993年,中俄列车遭暴徒洗劫强奸140个小时,邓公批示:跨国抓捕

旧闻档案
2023-09-27 21:39:31
全球官宣!5月7日,华为“亮剑”

全球官宣!5月7日,华为“亮剑”

互联鱼
2024-05-01 12:50:22
浙大邵逸夫医院楼志峰医生突然离世,终年仅51岁,值得引起重视!

浙大邵逸夫医院楼志峰医生突然离世,终年仅51岁,值得引起重视!

今日养生之道
2024-05-02 08:02:52
有种“整容”叫白百何换发型,本以为会很丑,结果惊为天人

有种“整容”叫白百何换发型,本以为会很丑,结果惊为天人

Yuki女人故事
2024-04-29 21:35:25
股市大事件!高层终于下狠手,比央妈降息还有爆发力,节后要爆了

股市大事件!高层终于下狠手,比央妈降息还有爆发力,节后要爆了

静守时光落日
2024-05-02 12:18:24
众生相!凯尔特人近8年第7次进次轮 双探花一脸淡定不庆祝

众生相!凯尔特人近8年第7次进次轮 双探花一脸淡定不庆祝

厝边人侃体育
2024-05-02 09:53:12
朱军的昔日搭档离世,曾令众人以为她离开央视,实则已故。

朱军的昔日搭档离世,曾令众人以为她离开央视,实则已故。

娱乐的小灶
2024-05-02 10:50:51
始料未及!英伟达正式宣布,外媒:这相当于是承认了

始料未及!英伟达正式宣布,外媒:这相当于是承认了

疯狂小菠萝
2024-05-02 11:13:43
2024-05-02 14:20:49
JAVA周某人
JAVA周某人
致敬大师,致敬未来的你。
137文章数 1513关注度
往期回顾 全部

科技要闻

4月交付量理想超问界,小米首月交付超七千

头条要闻

990万拍下迈巴赫还没付全款 二手车商"三戏"周鸿祎

头条要闻

990万拍下迈巴赫还没付全款 二手车商"三戏"周鸿祎

体育要闻

这场欧冠,多特的他诠释了什么叫"台柱子"

娱乐要闻

黄子韬被曝求婚徐艺洋 大量亲密照曝光

财经要闻

日元如此疯狂波动,背后谁在收割?

汽车要闻

预售2.89-3.49万 奔腾小马正式开启预售

态度原创

房产
艺术
本地
公开课
军事航空

房产要闻

单价2万内,装标4200+,主城改善大盘无套路硬刚!

艺术要闻

造科幻之物于园林 “天工开悟——夏航雕塑展”于南池子美术馆呈现

本地新闻

食味印象 | 潍坊:碳水脑袋的人间乐园

公开课

父亲年龄越大孩子越不聪明?

军事要闻

媒体:福建舰开始海试 最快正式服役可能要到2026年

无障碍浏览 进入关怀版