spring AOP(3)

spring AOP(3)

2210发表于2016-11-15

在前面我们讨论了如何编写通知,但是我们却没有指明在应用系统的什么地方应用这些通知。这就需要引入AOP中第4个重要的概念:切入点。切入点决定了一个特定类的特定方法是否满足一条特定的规则。如果一个方法确实符合,通知就应用到该方法上。spring的切入点可以让我们以一种灵活的方式定义在什么地方将通知织入到我们的类中。Spring已经提供了预定义的切入点实现。那么,在理解它之前,我们先要理解另一个AOP概念(概念真多,呵呵!):Advisor.也就是俗称的切面。大多数切面是由定义切面行为的通知和定义切面在什么地方执行的切入点组合而成。Spring认识到这一点,提供了Advisor,它把通知和切入点组合到一个对象中。精确地将,应该叫PointcutAdvisor。(大多数Spring自带的切入点都有一个对应的PointcutAdvisor),我们下面看到的,都是Spring的静态切入点。


<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">
    <!-- 各种通知 -->
    <!-- 前置通知 -->
    <bean id="myBeforeAdvice"
        class="com.wepull.spring.book.MyBeforeAdvice">
    </bean>
    <!-- 后置通知 -->
    <bean id="myAfterAdvice"
        class="com.wepull.spring.book.MyAfterAdvice">
    </bean>
    <!-- 异常通知 -->
    <bean id="myThrowsAdvice"
        class="com.wepull.spring.book.MyThrowsAdvice">
    </bean>
    <!-- 环绕通知 -->
    <bean id="myAroundAdvice"
        class="com.wepull.spring.book.MyAroundAdvice">
    </bean>
    <!-- 目标对象 -->
    <bean id="buyBookTarget"
        class="com.wepull.spring.book.MyBuyBook">
    </bean>
    <!-- 代理对象 -->
    <bean id="buyBook"
        class="org.springframework.aop.framework.ProxyFactoryBean">
        <!-- 实现的接口 -->
        <property name="proxyInterfaces">
            <value>com.wepull.spring.book.BuyBook</value>
        </property>
        <!-- 所有应用(通知)拦截器的名字 -->
        <property name="interceptorNames">
            <list>
                <!--
                    <value>myBeforeAdvice</value>
                    <value>myAfterAdvice</value>
                    <value>myThrowsAdvice</value>
                    <value>myAroundAdvice</value> -->
                <value>pointcutAdvisor1</value>
                <value>pointcutAdvisor2</value>
            </list>
        </property>
        <!-- 代理的目标对象 -->
        <property name="target">
            <ref bean="buyBookTarget" />
        </property>
    </bean>
 
    <!-- 配置前置通知应用的切入点 -->
    <bean id="pointcutAdvisor1"
        class="org.springframework.aop.support.NameMatchMethodPointcutAdvisor">
        <property name="mappedNames">
        <value>buy*</value>
        </property>
        <property name="advice" ref="myBeforeAdvice">
        </property>
    </bean>
    <!-- 配置后置通知应用的切入点 -->
    <bean id="pointcutAdvisor2"
        class="org.springframework.aop.support.NameMatchMethodPointcutAdvisor">
        <property name="mappedNames">
        <list>
                <value>buyBook</value>
                <value>buyBook2</value>
        </list>
        </property>
        <property name="advice" ref="myAfterAdvice">
        </property>
    </bean>
</beans>


 

当代理创建后,任何对我们的buyBook对象的以buy开头的方法调用都会添加myBeforeAdvice通知,对buyBook对象的buyBook和buyBook2方法调用添加myAfterAdvice通知。使用Namedmethodmatcher可以很清楚地指明哪些方法需要使用通知。然而,这样还是不能非常细致地控制切入点。为了更加精细的控制,我们来使用正则表达式切入点。


<!-- 配置前置通知应用的切入点 -->
    <bean id="pointcutAdvisor1"
        class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
        <property name="pattern">
        <value>.*buy.+k</value>
        <!--
        <value>com/.wepull/.spring/.book/.BuyBook/..*buy.*</value>
         -->
        </property>
        <property name="advice" ref="myBeforeAdvice">
        </property>
    </bean>


 

这个正则表达式的意思,就是我们的切入点应该匹配的方法是:类中以buy开头,后面至少有一个字符,最后结尾字符是k的方法。呵呵,正如大家看到的,它是一种比NameMatchMethodPointcut更加有表达力的定义切入点的方式。

 

聪明的大家已经观察到前面我们都是使用ProxyFactoryBean来创建我们的代理对象,这在没有很多类需要通知的小型系统中是可行的。但是如果有大量的类需要通知时,显式地在配置文件上创建每个代理就会显得很笨拙。幸运的是,Spring有一个自动代理机制,它可以让容器为我们产生代理。Spring中有两个类可以提供这种服务。你可以选择其中一种来实现,我们一起看看它们的配置:

1.     BeanNameAutoProxyCreator


<!-- 自动代理对象 -->
    <bean id="buyBook"
        class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
        <!-- 所有应用(通知)拦截器的名字 -->
        <property name="interceptorNames">
            <list>
                <value>pointcutAdvisor1</value>
            </list>
        </property>
        <property name="beanNames">
            <list>
                <value>buyBookTarget</value>
            </list>
        </property>
    </bean>


 

2.     DefaultAdvisorAutoProxyCreator


<bean id="autoProxyCreator"class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"></bean>


 

大家注意,如果你使用自动代理,你获取bean的时候要直接获取目标对象,通知已经自动织入到方法调用中。这样,我们就可以自动为很多目标对象创建通知,配置量也就大大减少了。除此以外,Spring还支持元数据(Metadata)自动代理,代理配置是通过源代码属性而不是外部配置文件获得的,非常强大。大家可以好生研究一下。呵呵,好啦,天色不早,我们今天就看到这里,祝大家学习愉快,咱们下次再会!

小编蓝狐