Spring AOP 定义切入点

首先我们编写了通知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]
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值