本文主要是介绍AOP创建切面,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
AOP,就是面向切面编程。
什么是切面呢?增强+切点就是切面。需要向切面里注入一个增强
前面说了增强,这里我们说一说切点,切点就是特定类的特定方法。
Pointcut = ClassFilter + MethodMatcher.
Advisor = Pointcut + Advice
三种切面类型:
- 一般切面
- 切点切面
- 引介切面
Advisor:一般切面,只包含增强,一般不会直接使用
PointcutAdvisor.切点切面,就是比较通用的Advice + Pointcut.
IntroductionAdvisor:引介切面。
切面的实现类
- 静态普通方法名匹配切面StaticMethodMatcherPointcutAdvisor.
匹配所有的目标类
package com.smart.advisor;import java.lang.reflect.Method;import org.springframework.aop.ClassFilter;
import org.springframework.aop.support.StaticMethodMatcherPointcutAdvisor;public class GreetingAdvisor extends StaticMethodMatcherPointcutAdvisor {public boolean matches(Method method, Class clazz) {return "greetTo".equals(method.getName());} public ClassFilter getClassFilter(){return new ClassFilter(){public boolean matches(Class clazz){return Waiter.class.isAssignableFrom(clazz);}};}
}
只需要实现matches方法,匹配所有的类。
覆盖getClassFilter().让他仅匹配waiter类极其子类。
我们还需要一个增强
package com.smart.advisor;import java.lang.reflect.Method;import org.springframework.aop.MethodBeforeAdvice;public class GreetingBeforeAdvice implements MethodBeforeAdvice {public void before(Method method, Object[] args, Object obj) throws Throwable {String clientName = (String)args[0];System.out.println(obj.getClass().getName()+"."+method.getName());System.out.println("How are you!Mr."+clientName+".");}
}
spring配置来定义切面
<!-- 普通方法名匹配切面 --><bean id="waiterTarget" class="com.smart.advisor.Waiter" /><bean id="sellerTarget" class="com.smart.advisor.Seller" /><bean id="greetingAdvice"class="com.smart.advisor.GreetingBeforeAdvice" /><!--切面中注入增强--><bean id="greetingAdvisor" class="com.smart.advisor.GreetingAdvisor"p:advice-ref="greetingAdvice" /><!--公共配置信息--><bean id="parent" abstract="true"class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="greetingAdvisor" p:proxyTargetClass="true" /><!--两个代理Bean通过parent放入代理工厂中--><bean id="waiter" parent="parent" p:target-ref="waiterTarget" /><bean id="seller" parent="parent" p:target-ref="sellerTarget" />
当有多个类似的方法,需要配置切面的时候,就应该使用正则表达式方法匹配切面。
RegexpMethodPointcutAdvisor
<!-- 正则表达式方法名匹配切面 --><bean id="regexpAdvisor"class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"p:advice-ref="greetingAdvice"> <!--增强注入--><property name="patterns"> <!--匹配多个正则方法--><list><value>.*greet.*</value></list></property></bean><bean id="waiter1" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="regexpAdvisor" <!--表明切面-->p:target-ref="waiterTarget" <!--表明目标类-->p:proxyTargetClass="true" />
动态切面
应用场景:比如服务员打招呼,只有当顾客是刘洋的时候,她才启用增强。
这就是动态界面,和方法的入参有关系。的切面
通过继承DynamicMethodMathcherPointcut类来实现。这个只是切点类。
通过配置,切点类+增强类 就有了切面。
package com.smart.advisor;import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;import org.springframework.aop.support.DynamicMethodMatcherPointcut;public class GreetingDynamicPointcut extends DynamicMethodMatcherPointcut {private static List<String> specialClientList = new ArrayList<String>();static {specialClientList.add("John");specialClientList.add("Tom");}
// public ClassFilter getClassFilter() {
// return new ClassFilter() {
// public boolean matches(Class clazz) {
// System.out.println("调用getClassFilter()对"+clazz.getName()+"做静态检查.");
// return Waiter.class.isAssignableFrom(clazz);
// }
// };
// }
// public boolean matches(Method method, Class clazz) {
// System.out.println("调用matches(method,clazz)对"+clazz.getName()+"."+method.getName()+"做静态检查.");
// return "greetTo".equals(method.getName());
// }public boolean matches(Method method, Class clazz, Object[] args) {System.out.println("调用matches(method,clazz)对"+clazz.getName()+"."+method.getName()+"做动态检查.");String clientName = (String) args[0];return specialClientList.contains(clientName);}}<!-- 动态切面 --><bean id="dynamicAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"><property name="pointcut"><bean class="com.smart.advisor.GreetingDynamicPointcut" /></property><property name="advice"><bean class="com.smart.advisor.GreetingBeforeAdvice" /></property></bean><bean id="waiter2" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="dynamicAdvisor" p:target-ref="waiterTarget"p:proxyTargetClass="true" />
流程切面:
前面的动态切面和入参有关,流程切面和调用者有关系。比如:只有通过service调用的greetto,和serveto才能触发他们的增强。通过ControlFlowPointcut实现。
package com.smart.advisor;public class WaiterDelegate {private Waiter waiter;public void service(String clientName) {waiter.greetTo(clientName);waiter.serveTo(clientName);}public void setWaiter(Waiter waiter) {this.waiter = waiter;}
}
通过spring配置切面:
<!-- 控制流程切面 --><bean id="controlFlowPointcut" class="org.springframework.aop.support.ControlFlowPointcut"><constructor-arg type="java.lang.Class"value="com.smart.advisor.WaiterDelegate" /><constructor-arg type="java.lang.String" value="service" /><!--service 所有的方法都会织入增强--></bean><bean id="controlFlowAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"p:pointcut-ref="controlFlowPointcut" p:advice-ref="greetingAdvice" /><bean id="waiter3" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="controlFlowAdvisor" p:target-ref="waiterTarget"p:proxyTargetClass="true" />
复合切点切面:
上面的流程切面,service中所有的方法都会织入增强,如果我们只想greeTO()织入增强,我们应该如何做呢?这时候就可以使用复合切点切面:流程切点+方法名切点
ComposablePointcut.提供了切点之前复合运算的功能。
通过Pointcuts工具类,可以完成切点的交集,并集运算。
intersection
union
package com.smart.advisor;
import org.springframework.aop.Pointcut;
import org.springframework.aop.support.ComposablePointcut;
import org.springframework.aop.support.ControlFlowPointcut;
import org.springframework.aop.support.NameMatchMethodPointcut;public class GreetingComposablePointcut {public Pointcut getIntersectionPointcut(){ComposablePointcut cp = new ComposablePointcut(); //复合切点Pointcut pt1 = new ControlFlowPointcut(WaiterDelegate.class,"service");//流程切点NameMatchMethodPointcut pt2 = new NameMatchMethodPointcut();//方法名切点pt2.addMethodName("greetTo");return cp.intersection(pt1).intersection((Pointcut)pt2); //注意这里的方法}
}
<!-- 复合切点切面 --><bean id="gcp" class="com.smart.advisor.GreetingComposablePointcut" /><bean id="composableAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"p:pointcut="#{gcp.intersectionPointcut}" p:advice-ref="greetingAdvice" /><bean id="waiter4" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="composableAdvisor" p:target-ref="waiterTarget"p:proxyTargetClass="true" />
引介切面:
是引介增强的封装器;
另外:
前面我们都通过proxyFactoryBean配置代理。Spring中我们使用了BeanPostProcessor 来完成这项工作。
实现类:
自动创建代理:根据名字找到代理的目标类,根据切面中的信息找到代理的目标类。
<!-- 通过Bean名称自动创建代理 --><bean
class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"p:beanNames="*er" p:interceptorNames="greetingAdvice"p:optimize="true"/>
<!--通过Advisor自动创建代理--><bean id="regexpAdvisor"class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"p:patterns=".*greet.*" p:advice-ref="greetingAdvice" /><bean
class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" p:proxyTargetClass="true" />
注意:目标类的内部方法调用是不会使用代理类处理的。比如serverto()中调用greetTo 的时候不会被增强、
这篇关于AOP创建切面的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!