![Spring 5企业级开发实战](https://wfqqreader-1252317822.image.myqcloud.com/cover/807/26542807/b_26542807.jpg)
3.4 基于Spring AOP的实战
3.4.1 增强类型
AOP联盟为增强定义了org.aopalliance.aop.Advice接口,Spring支持5种类型的增强。本章3.3节中使用到的@Before、@After等注解是基于AspectJ实现的增强类型。其实Spring也支持很多增强类型,Spring AOP按照增强在目标类方法中的连接点位置可以分为5种。
• 前置增强:表示在目标方法执行前实施增强。
• 后置增强:表示在目标方法执行后实施增强。
• 环绕增强:表示在目标方法执行前后实施增强。
• 异常抛出增强:表示在目标方法抛出异常后实施增强。
• 引介增强:表示在目标类中添加一些新的方法和属性。
以下将依次介绍每种增量类型,由于基于XML和基于注解的配置其本质都是相同的,因此下面将只通过注解的方式演示Spring各种增强类型,并观察各种增强类型的执行时序。
3.4.2 前置增强
Spring的前置增强主要接口是MethodBeforeAdvice,其顶级接口是AOP联盟中的Advice接口。从如图3-6所示的类图可以发现,Spring的前置增强扩展了Advice接口。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P86_94677.jpg?sign=1738864277-gOYn80Diw97psz1qVOLOumVTI6pX1ANm-0-7ed7cd60dd84740f6b89268fe2fd617e)
图3-6 MethodBeforeAdvice相关类图
下面将通过实现MethodBeforeAdvice接口来新增一个前置增强实现类,然后通过案例阐述使用Spring的前置增强类型的编程方式。前置增强实现类的代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P86_94675.jpg?sign=1738864277-Zg1LBcY37h9i7T4inghZ1ZoUeQ53mXpU-0-70c5c93ee8c7a1fcb07a7ada8777ec01)
下面将创建一个被增强类Waiter,用于被SpringBeforeAdvice类增强,Waiter类的代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P86_94676.jpg?sign=1738864277-vtR3ZHtaZxxd9p9VnldgZSUfNelmxr4m-0-76f2f8b0b5cb526e963b4c24c5842b17)
测试代码中需要创建被代理对象和前置增强的对象,并通过Spring生成代理对象,测试代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P87_94679.jpg?sign=1738864277-DoxhM23qTSxXix9Sjo53AkyfVAiUcIZS-0-97ab78fd0db620979adee15072bdbdd2)
ProxyFactory生成的代理对象proxy就是被增强后的对象,运行测试代码,得到的运行结果如图3-7所示。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P88_29622.jpg?sign=1738864277-OiatzGp4hQKJXdnDYbIRTyhcqOksYCqD-0-36f4dedea804e77cf991b0aaf9f1b470)
图3-7 前置增强测试效果图
由测试结果可以看出,在Waiter类的serve方法执行之前,前置增强的逻辑执行了。前置增强即在目标方法执行前实施增强逻辑。
3.4.3 后置增强
Spring的后置增强主要接口是AfterReturningAdvice,其类图如图3-6所示。
下面将通过实现AfterReturningAdvice接口来新增一个后置增强实现类,然后通过3.4.2节中的Waiter案例阐述使用Spring的后置增强类型的编程方式。后置增强实现类的代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P88_94680.jpg?sign=1738864277-DVem5Mnn5Qrj18gQDV7wdX77Cgek1kow-0-7359ce93c97ac07db912e1255ce64088)
测试代码中只需修改一行代码:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P88_94681.jpg?sign=1738864277-wDeXEhB9Ro7bEvV3wXnrFqoonnkKfmpQ-0-4fddb7a2aa6dd2b4c395c052ec4f52d9)
测试结果如图3-8所示。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P89_29771.jpg?sign=1738864277-rjxPqbnkt7S1U4axZiZZGBadsFjqlhPg-0-0af9c02c6499138eceb4d648a5567dcf)
图3-8 后置增强测试效果图
3.4.4 环绕增强
Spring的环绕增强主要接口是MethodInterceptor,其类图如图3-6所示。
下面将通过实现MethodInterceptor接口来新增一个环绕增强实现类,然后通过3.4.2节中的Waiter案例阐述使用Spring环绕增强类型的编程方式。环绕增强实现类的代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P89_94683.jpg?sign=1738864277-uytndlyvnzXM3frKQQ47lpCxx1MDmLUF-0-2fb535398665899e93c002fc9f1875f8)
测试代码只需要修改使用环绕增强实现类SpringMethodInterceptor,测试结果如图3-9所示。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P89_29892.jpg?sign=1738864277-WwySbT7tHRKoTurcN7MpctzTyWwIZzKW-0-464b116dabf71214b487073234af21e4)
图3-9 环绕增强测试效果图
3.4.5 异常抛出增强
Spring的异常抛出增强主要接口是ThrowsAdvice,其类图如图3-6所示。
下面将通过实现ThrowsAdvice接口来新增一个异常抛出增强实现类,然后通过3.4.2节中的Waiter案例阐述使用Spring的异常抛出增强类型的编程方式。异常抛出增强实现类的代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P90_94684.jpg?sign=1738864277-Bu9JdNBxohA6V5rZDcCLbdmMDOZlYlaq-0-d972920dd87e800ea3bda77a011b850c)
异常抛出增强的测试代码执行效果如图3-10所示。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P90_29978.jpg?sign=1738864277-tyMxyKBgYbzucAWJSsTkslDuWRKOpQdH-0-0bf680b2b47784845b18c38b12dab57e)
图3-10 异常抛出增强测试效果图
3.4.6 引介增强
引介增强的目标是在目标类中添加一些新的方法和属性。以Waiter类为例,现在想给其添加一个Management接口中的manage()方法而不修改Waiter类的代码。Management代码如下所示:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P90_94685.jpg?sign=1738864277-i7cycsTSaXQop1Of21d8Gq6PGIBTJfwu-0-dedae33a27b7e0c46ebb765af1099c6f)
Spring的引介增强主要接口是IntroductionInterceptor,通过图3-11可以看出,Spring已经提供了IntroductionInterceptor接口的实现类DelegatingIntroductionInterceptor。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P91_94690.jpg?sign=1738864277-AyXJFNwGibgzOWCdDZOXFzpDl8YAaqPZ-0-bb07a6cf820ff638c5d5df825aa746f9)
图3-11 IntroductionInterceptor相关类图
下面将通过扩展DelegatingIntroductionInterceptor来实现引介增强。通过Manager类继承DelegatingIntroductionInterceptor并实现Management接口,Manager代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P91_94687.jpg?sign=1738864277-MMCdIFCFOjug6KyZswWLOoWr0mE12WYh-0-a498f45445718053916c8ce5982fc706)
此时需要修改配置文件,需要指定引介增强所在的实现接口并需要将proxyTargetClass属性设置为true。具体配置文件如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P91_94688.jpg?sign=1738864277-fDNmkugF36FFcvPpVDrxFGkZzp9ws4wb-0-9d0e683b2c3217f16f687cb8e9a8ea99)
测试代码中,从Spring上下文中获取代理对象waiterProxy,将其强制转化为一个Management对象。修改后测试代码如下:
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P92_94692.jpg?sign=1738864277-ihXqideW6gNg0MLyn56WiuEooYi1e4pb-0-e76a6aec4be794e84514c6bf04395c22)
运行测试结果如图3-12所示,发现Waiter类的代理对象多了一个新的功能,可以调用Management接口的manage方法。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P92_30313.jpg?sign=1738864277-ZGuErc2OVnd31WLdk42nuXM0rlGdOf1N-0-9192fab4efca8980d8a7ce0d551d3a6d)
图3-12 引介增强测试效果图
3.4.7 切入点类型
如3.2.2节所述,切入点是匹配连接点的拦截规则。之前的案例中使用的是注解@Pointcut,该注解是AspectJ中的。除了这个注解之外,Spring也提供了其他一些切入点类型:
• 静态方法切入点StaticMethodMatcherPointcut
• 动态方法切入点DynamicMethodMatcherPointcut
• 注解切入点AnnotationMatchingPointcut
• 表达式切入点ExpressionPointcut
• 流程切入点ControlFlowPointcut
• 复合切入点ComposablePointcut
• 标准切入点TruePointcut
各种切入点的类图如图3-13所示。
![](https://epubservercos.yuewen.com/45D01C/15056703904176006/epubprivate/OEBPS/Images/Figure-P93_30327.jpg?sign=1738864277-6QZ2L0HoTMZlxvMpIfAbz4Ibes2erF0F-0-849e60f2653ecd79827293a04269d1c9)
图3-13 切入点各类的类图