首先我们编写了通知advice,但是我们还不能表达在应用系统的什么地方应用这些通知,切入点决定了一个特定类的特定方法是否满足特定规则,如果满足则通知就应用到该方法上,Spring的切入点可以让我们灵活的定义在什么地方应用通知。

Spring的切入点框架的核心接口PointCut

public interface PointCut {
  ClassFilter getClassFilter();
  MethodMatcher getMethodMatcher();
}




PointCut 是根据方法和类决定在什么地方织入通知的。



ClassFilter决定了一个类是否符合通知的要求


public interface ClassFilter{
   boolean matches(Class clazz);//根据类名判断
}



实现了这个接口的类决定了以参数传进来的类是否应该被通知,它相当于一个类的过滤器,一般根据类名过滤,另外这个接口总是包含一个简单的ClassFilter的实现ClassFilter.TRUE,它是规范任何类的ClasFilter实例,它适用于只创建只根据方法来决定是否是应用通知的切入点。



MethodMether决定了一个类的一个方法是否符合通知的要求


public interface MethodMether{
   //决定一个类的一个方法是否被通知,AOP代理被创建的时候,调用一次这个方法,这个方法的结果决定了是否应用通知
   boolean mathces(Method m,Class target);
  //决定MethodMether是静态还是动态,静态:false,通知总是被执行,动态true:根据运行时方法的参数值决定通知是否被执行。
   public boolean isRuntime();
   //如果是静态切入点,此方法不会被调用
   //如果是动态切入点,目标对象方法每次被调用的时候,此方法被调用。
   public boolean matches(Method m,Class target,Object arg[]);
}




Advisor



大多数的切面由通知和切入点组成,因此Spring把advice和pointcut组合为一个对象,PointcutAdvisor


public interface PointcutAdvisor{
   PointCut getPointCut();
   Advice getAdvice();
}
//大多数的Spring自带的切入点都有一个对应的PointcutAdvisor.




静态/动态切入点比较



静态切入点只在代理被创建的时候执行一次,而不是在运行期间每次方法调用都执行,因此性能比动态切入点好,因此静态切入点是我们的首选,Spring为创建静态切入点提供了父类StaticMethodMatcherPointcut,继承它并实现isMatch方法就可以了



Spring提供的静态切入点



(1)NameMatchMethodPointcut


这个类有2个主要方法:


public void setMappedName(String name)
   public void setMappedNames(String[] names)



事例:


有一个接口类 MyInterfaceA ,包含3个方法,set1,set2,get3 ,有一个实现类MyClassAImpl;我们想配置set方法的切入点,如下实现:


<beans>
    //目标对象
    <bean id="MyClassATarget" class="MyClassAImpl"></bean>
    //通知
    <bean id="myAdvice" class="..."></bean>
    //定义切面
    <bean id="myAdvisor" class="....NameMatchMethodPointcut">
          //方法过滤,注入切入点
          <property name="mappedName">
               <value>set*</value>
          </property>
          //注入通知
          <property name="advice">
               <ref bean="myAdvice"/>
          <property>
    </bean>
   //配置代理
    <bean id="MyClassAProxy" class=".....ProxyFactoryBean">
       //代理的接口
       <property name="proxyInterfaces">
           <value>MyInterfaceA</value>
       <property>
       //配置切面
       <property name="interceptorNames">
           <list>
               <value>myAdvisor</value>
           </list>
       <property>
       //配置目标对象
       <property name="target">
           <ref bean="MyClassATarget"/>
       <property>
    </bean>
 </beans>

//整个过程可以如下几步:
1  编写接口:myinterface
2  编写接口的实现myImpl并配置为bean:myImplTarget   
3  编写通知myAdvice并配置为bean:myAdvice
4  配置切面myAdvisor,切面中注入切入点(set*)和通知myAdvice
5  配置代理myImplProxy,注入接口类myinterface,注入切面myAdvisor,注入目标对 象myImplTarget





(2)RegexpMethodPointcut正则表达式切入点


整个过程同上,但在配置切面时有所不同,见蓝色部分代码


//定义切面
    <bean id="myAdvisor" class="....RegexpMethodPointcut">
          //方法过滤,注入切入点
          [color=blue]<property name="pattern">
               <value>.*get.+By.+</value>
          </property>[/color]          //注入通知
          <property name="advice">
               <ref bean="myAdvice"/>
          <property>
    </bean>



[color=blue]正则表达式:


.:匹配任何单个字符 例如:setF. 匹配setFi,但不匹配setF 和 setFii


+:匹配前一个字符一次或者多次,例如setF.+ 匹配setFBar和setFB,不匹配setF


*:匹配前一个字符0次或者多次,例如setF.* ,同上并匹配setF


\:匹配任何正则表达式符号,例如\.setF ,匹配bar.setF ,不匹配setF[/color]