22_AOP_切面——静态切面
【Spring AOP 如何定位连接点】
1.增强提供了连接点的方位信息:如织入到方法前面、后面等。
2.切点描述的是织入到哪些类的哪些方法上。
【切点】
Spring通过org.springframework.aop.Pointcut接口描述切点,Pointcut由ClassFilter和MethodMatcher构成,通过ClassFilter定位到某些特定的类,通过MethodMatcher定位到某些特定的方法。这样,Pointcut就拥有了描述某些类的某些特定方法的能力。
可以看到:
ClassFilter只定义了一个方法matches(Class clazz),其参数代表一个被检测的类,该方法判别被检测的类是否匹配过滤条件。
Spring提供了两种方法匹配器:静态方法匹配器和动态方法匹配器。
1.静态方法匹配器
仅对方法名签名(包括方法名和入参类型及顺序)进行匹配。静态匹配仅判断一次。
2.动态方法匹配器
会在运行期间检查方法入参的值。动态匹配因为每次调用方法的入参可能不一样,导致每次调用方法都必须判断,因此动态匹配对性能的影响较大。
一般情况,动态匹配不常使用。
方法匹配器的类型由MethodMatcher接口的isRuntime()方法决定,返回false表示是静态方法匹配器,返回true表示是动态方法匹配器。
【切点类型】
Spring提供了6种切点:
1.静态方法切点:org.springframework.aop.support.StaticMethodMatcherPointcut
StaticMethodMatcherPointcut是静态方法切点的抽象基类,默认情况下匹配所有的类。StaticMethodMatcherPointcut有两个重要的子类:NameMethodMatcherPointcut和AbstractRegexMethodPoint。前者提供简单的字符串匹配方法签名,后者使用正则表达式匹配方法签名。
2.动态方法切点:org.springframework.aop.support.DynamicMethodMatcherPointcut
DynamicMethodMatcherPointcut是动态方法切点的抽象基类,默认情况下它匹配所有的类。DynamicMethodMatcherPointcut已过时!!使用DefaultPointcutAdvisor和DynamicMethodPointcut动态方法匹配器代替。
3.注解切点
4.表达式切点
5.流程切点
6.复合切点
【切面类型】
Spring使用org.springframework.aop.Advisor接口表示切面的概念。
一个切面同时包含横切代码和连接点信息。切面分为三类:一般切面、切点切面、引介切面。
1.一般切面:Advisor
它仅包含一个Advice,Advice包含了横切代码和连接点的信息,所以Advice本身就是一个简单的切面,只不过它代表的是所有目标类的所有方法。由于这个横切面过于宽泛,所以一把不会直接使用。
2.切点切面:PointcutAdvisor
包含Advice和Pointcut两个类。我们可以通过类、方法名以及方法方位等信息灵活定义切面的连接点,提供更具适用性的切面。
3.引介切面:IntroductionAdvisor
引介切面是对应引介增强的特殊的切面,它应用于类层面之上,所以引介切点适用ClassFilter进行定义。
【静态普通方法名匹配切面 例子】
【奔驰车类:BenzCar.java】
package com.Higgin.part4; public class BenzCar { public void driving(){ System.out.println("奔驰车在行驶..."); } }
【宝马车类:BMWCar.java】
package com.Higgin.part4; public class BMWCar { public void driving(){ System.out.println("宝马车在行驶..."); } }
【汽车切面类:CarAdvisor.java】
package com.Higgin.part4; import java.lang.reflect.Method; import org.springframework.aop.ClassFilter; import org.springframework.aop.support.StaticMethodMatcherPointcutAdvisor; /** * 汽车类的切面CarAdvisor类 * 实现接口:StaticMethodMatcherPointcutAdvisor * 实现StaticMethodMatcherPointcutAdvisor接口唯一需要定义的是matches()方法, * */ public class CarAdvisor extends StaticMethodMatcherPointcutAdvisor{ /** * 切点方法 匹配 * 匹配规则:方法名为driving * 默认情况下,匹配所有的类 */ @Override public boolean matches(Method method, Class<?> clazz) { return "driving".equals(method.getName()); } /** * 通过覆盖getClassFilter()方法,让它仅匹配BenzCar类及其子类 */ public ClassFilter getClassFilter(){ return new ClassFilter() { /** * 切点类 匹配 * 匹配规则:为BenzCar类或其子类 */ @Override public boolean matches(Class<?> clazz) { return BenzCar.class.isAssignableFrom(clazz); } }; } }
【driving方法的前置增强类:DrivingBeforeAdvice.java】
package com.Higgin.part4; import java.lang.reflect.Method; import org.springframework.aop.MethodBeforeAdvice; /** * Driving的前置增强类 */ public class DrivingBeforeAdvice implements MethodBeforeAdvice{ @Override public void before(Method method, Object[] args, Object obj) throws Throwable { System.out.println("要增强的是:"+obj.getClass()+"类 ---"+method.getName()+"方法"); //得到切点的信息 System.out.println("【前置增强】做好行驶前的准备工作..."); } }
【Spring的xml配置文件:part4.xml】
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:p="http://www.springframework.org/schema/p" xmlns:util="http://www.springframework.org/schema/util" xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:cache="http://www.springframework.org/schema/cache" xsi:schemaLocation=" http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-3.1.xsd http://www.springframework.org/schema/cache http://www.springframework.org/schema/cache/spring-cache-3.1.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd"> <!-- 要增强的目标对象1 --> <bean id="benzTarget" class="com.Higgin.part4.BenzCar"/> <!-- 要增强的目标对象2 --> <bean id="bmwTarget" class="com.Higgin.part4.BMWCar"/> <!-- 前置增强 --> <bean id="drivingBeforeAdvice" class="com.Higgin.part4.DrivingBeforeAdvice" /> <!-- 切面 --> <bean id="carAdvisor" class="com.Higgin.part4.CarAdvisor" p:advice-ref="drivingBeforeAdvice"/> <!-- Spring代理工厂的成员变量配置 --> <bean id="parent" abstract="true" class="org.springframework.aop.framework.ProxyFactoryBean" p:interceptorNames="carAdvisor" p:proxyTargetClass="true" /> <!-- Benz代理 --> <bean id="benz" parent="parent" p:target-ref="benzTarget" /> <!-- BMW代理 --> <bean id="bmw" parent="parent" p:target-ref="bmwTarget" /> </beans>
【测试类:TestCarAdvisor.java】
package com.Higgin.part4.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import com.Higgin.part4.BMWCar; import com.Higgin.part4.BenzCar; public class TestCarAdvisor { public static void main(String[] args) { ApplicationContext context=new ClassPathXmlApplicationContext("part4.xml"); BenzCar benz= (BenzCar) context.getBean("benz"); BMWCar bmw=(BMWCar) context.getBean("bmw"); benz.driving(); //奔驰车的driving方法 System.out.println("==========================="); bmw.driving(); //宝马车的driving方法 } }
【运行结果】
【静态正则表达式方法 匹配切面 例子】
【奔驰车类 BenzCar.java】
package com.Higgin.part5; public class BenzCar { public void driving(){ System.out.println("benz车行驶....."); } public void stop(){ System.out.println("benz车停止....."); } public void sliding(){ System.out.println("benz车漂移....."); } }
【前置增强类 】
package com.Higgin.part5; import java.lang.reflect.Method; import org.springframework.aop.MethodBeforeAdvice; /** * 汽车类的前置增强类 */ public class CarBeforeAdvice implements MethodBeforeAdvice{ @Override public void before(Method method, Object[] args, Object obj) throws Throwable { System.out.println("要增强的是:"+obj.getClass()+"类 ---"+method.getName()+"方法"); //得到切点的信息 System.out.println("【前置增强】做好行驶前的准备工作..."); } }
【Spring的xml匹配 part5.xml】
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:p="http://www.springframework.org/schema/p" xmlns:util="http://www.springframework.org/schema/util" xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:cache="http://www.springframework.org/schema/cache" xsi:schemaLocation=" http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-3.1.xsd http://www.springframework.org/schema/cache http://www.springframework.org/schema/cache/spring-cache-3.1.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd"> <!-- 要增强的目标对象1 --> <bean id="benzTarget" class="com.Higgin.part5.BenzCar"/> <!-- 前置增强 --> <bean id="drivingBeforeAdvice" class="com.Higgin.part5.CarBeforeAdvice" /> <!-- 正则表达式 匹配 --> <bean id="regexAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" p:advice-ref="drivingBeforeAdvice"> <property name="patterns"> <list> <value>.*ing.*</value> </list> </property> </bean> <!-- Spring代理工厂的成员变量配置 --> <bean id="benzCar" class="org.springframework.aop.framework.ProxyFactoryBean" p:interceptorNames="regexAdvisor" p:target-ref="benzTarget" p:proxyTargetClass="true" /> </beans>
【测试类】
package com.Higgin.part5.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import com.Higgin.part5.BenzCar; public class TestRegexAdvisor { public static void main(String[] args) { ApplicationContext context=new ClassPathXmlApplicationContext("part5.xml"); BenzCar benz=(BenzCar) context.getBean("benzCar"); benz.driving(); //匹配.*ing.* System.out.println("==================================="); benz.sliding(); //匹配.*ing.* System.out.println("==================================="); benz.stop(); //不匹配 } }
【运行结果】