亚洲国产日韩欧美一区二区三区,精品亚洲国产成人av在线,国产99视频精品免视看7,99国产精品久久久久久久成人热,欧美日韩亚洲国产综合乱

學習AOP之深入一點Spring Aop

Original 2016-11-12 14:01:32 723
abstract:上篇中最后有那段代碼使用了一個ProxyFactory類來完成代理的工作,從而實現(xiàn)了Aop的Around Advice,代碼如下:package aop.demo; import org.springframework.aop.framework.ProxyFactory; public class ClientCode {   

上篇中最后有那段代碼使用了一個ProxyFactory類來完成代理的工作,從而實現(xiàn)了Aop的Around Advice,代碼如下:

package aop.demo;
import org.springframework.aop.framework.ProxyFactory;
public class ClientCode {
    public static void main(String[] args) {
        ProxyFactory proxyFactory = new ProxyFactory();     // 創(chuàng)建代理工廠
        proxyFactory.setTarget(new SayImpl());         // 射入目標類對象
        proxyFactory.addAdvice(new SayImplAroundAdvice());
        ISay say = (ISay) proxyFactory.getProxy();
        say.say();
    }
}

那么接下來就聊聊ProxyFactory吧,看看它都干了些啥。
1、ProxyFactory的奧秘


繼續(xù)看上面的代碼只用了5行,這里面意思也非常明確,只有在第4行的時候有一個getProxy的方法并轉(zhuǎn)換為ISay接口??磥泶韺ο蟮膩碓纯梢詮乃胧至?。

public Object getProxy() {
    return createAopProxy().getProxy();
}

只不過代碼只有一行,調(diào)用的是一個createAopProxy()的方法返回了AopProxy類型的對象,再通過AopProxy的getProxy來獲得了代理對象。

那么只好再看一下createAopProxy()是啥樣子咯:

protected final synchronized AopProxy createAopProxy() {
    if (!this.active) {
        activate();
    }
    return getAopProxyFactory().createAopProxy(this);
}

這個方法在org.springframework.aop.framework.ProxyCreatorSupport這個類里面,ProxyFactory是繼承了它的。這個類字面意思就是一個代理創(chuàng)建的支持類。

但是看了createAopProxy方法后又郁悶了,還有一個getAopProxyFactory(),真是一層套一層啊。當然這里還是需要從類的層次結構來看會清楚一些,只是我主要是看它是怎么生成代理對象的,設計上的事情回頭再看。

//這個方法訪問了一個內(nèi)部成員
public AopProxyFactory getAopProxyFactory() {
    return this.aopProxyFactory;
}
//再看aopProxyFactory其實是在構造函數(shù)里創(chuàng)建的
public ProxyCreatorSupport() {
    this.aopProxyFactory = new DefaultAopProxyFactory();
}

這里看到了DefaultAopProxyFactory這個工廠類,好了,也就是它才是創(chuàng)建代理的真正人物。那么這里接著createAopProxy直接看代碼:

public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
    if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
        Class targetClass = config.getTargetClass();
        if (targetClass == null) {
            throw new AopConfigException("TargetSource cannot determine target class: " +
                    "Either an interface or a target is required for proxy creation.");
        }
        if (targetClass.isInterface()) {
            return new JdkDynamicAopProxy(config);
        }
        return CglibProxyFactory.createCglibProxy(config);
    }
    else {
        return new JdkDynamicAopProxy(config);
    }
}

從這里可以看到有兩種AopProxy的代理:Cglib和Jdk。它們倆的區(qū)別:

    Cglib創(chuàng)建代理慢但執(zhí)行快而且可以代理類
    Jdk創(chuàng)建代理快但執(zhí)行慢,只可以代理接口

順著ClientCode這個代碼肯定是用JdkDynmaicAopProxy,最終proxyFactory.getProxy()調(diào)用的是JdkDynmaicAopProxy的實例。那好就看一下JdkDynmaicAopProxy中getProxy都做了啥吧:

public Object getProxy() {
    return getProxy(ClassUtils.getDefaultClassLoader());
}
public Object getProxy(ClassLoader classLoader) {
    if (logger.isDebugEnabled()) {
        logger.debug("Creating JDK dynamic proxy: target source is " + this.advised.getTargetSource());
    }
    Class[] proxiedInterfaces = AopProxyUtils.completeProxiedInterfaces(this.advised);
    findDefinedEqualsAndHashCodeMethods(proxiedInterfaces);
    return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);
}

好了,這里看到了熟悉的代碼,即通過Proxy.newProxyInstance生成代理對象交給調(diào)用者。Spring通過抽象工廠模式設計了兩種代理方法。
2、再看看ProxyFactroyBean


但是在xml配置的時候用的并不是ProxyFactory,而是ProxyFactroyBean。有點奇怪,為什么會有兩個類呢?先來看看ProxyFactoryBean:

public class ProxyFactoryBean extends ProxyCreatorSupport
        implements FactoryBean<Object>, BeanClassLoaderAware, BeanFactoryAware {

哦喲,原來這家伙繼承了FactoryBean,好了,原來它借且FactoryBean提供了一個中間層。因為Spring如果發(fā)現(xiàn)Ioc創(chuàng)建的對象帶有FactoryBean時會調(diào)用FactoryBean的getObject方法來獲得對象。有了這個它在Ioc容器里獲得的便是getObject返回的代理對象,而不是返回ProxyFactoryBean本身,這樣才能注入嘛。所以ProxyFactoryBean主要是使用在Ioc容器里的。

具體getObject里實現(xiàn)的原理和ProxyFactory類似,主要還是和ProxyCreatorSupport有關,ProxyCreatorSupport封裝了這部分邏輯,所以可以復用。
3、進入切面的小世界


寫了這么多發(fā)現(xiàn)我還是停留在“代理”的層面,但是AOP難道僅僅止于此嗎?當然不是,比如ISay接口增加一個noaop方法,這個方法我就不希望被代理,那怎么做呢?

先調(diào)整一下例子代碼,增加noaop方法。

public interface ISay {
    void say();
    void noaop();
}
public class SayImpl implements ISay{
    public void say() {
        System.out.print("我是5207.");
    }
    public void noaop() {
        System.out.println("別aop我");
    }
}

好了,然后增加一個切面,讓這個切面去做分辨,以xml配置為例,下面對spring.xml做一下修改:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context.xsd">
    <!-- 聲明被代理的目標對象 -->
    <bean id="sayImpl" class="aop.demo.SayImpl"></bean>
    <!-- 聲明用于增強的攔截器對象 -->
    <bean id="sayImplAroundAdvice" class="aop.demo.SayImplAroundAdvice"></bean>
    <!-- 配置一個切面 -->
    <bean id="sayAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
        <property name="advice" ref="sayImplAroundAdvice"/>            <!-- 增強 -->
        <property name="pattern" value="aop.demo.SayImpl.s.*"/> <!-- 切點(正則表達式) -->
    </bean>   
    
    <!-- 聲明代理對象 -->
    <bean id="sayProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
        <property name="interfaces" value="aop.demo.ISay"/>            <!-- 這個就是被代理的接口 -->
        <property name="target" ref="sayImpl"/>                        <!-- 這個就是被代理的對象 -->
        <property name="interceptorNames" value="sayAdvisor"/>         <!-- 這個就是代理的增強器 --> 
    </bean>
</beans>

上面xml中新增了一個切面sayAdvisor,它的作用是以正則表達式的規(guī)則來選擇是否aop。比如本例子的意思是只代理SyaImpl的s開頭的方法。那么noaop方法應該是不會被代理啦。

client代碼也修改一下:

public class Client {
    @SuppressWarnings("resource")
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("aop/demo/spring.xml");
        ISay say = (ISay)context.getBean("sayProxy");
        say.say();
        say.noaop();//增加noaop的調(diào)用,看看會不會被代理
    }
}

執(zhí)行的結果如下:

    大家好:我是5207.希望大家多多點贊.
    別aop我

這就發(fā)現(xiàn)advisor已經(jīng)有效果啦。
4、自動完成對切面的代理


之前的各種代碼都帶有一個問題,就是client最終調(diào)用的時候都是獲得的代理對象,如下面的代碼:

ISay say = (ISay)context.getBean("sayProxy");

那在做Aop增強的時候改老的代碼,這樣就失敗了Aop的意義了。所以沒有辦法可以自動就完成這個操作,只要配置好就可以透明的完成這個代理過程呢?

spring提供了自動代理的實現(xiàn),對spring.xml做一下調(diào)整:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context.xsd">
    <!-- 聲明被代理的目標對象 -->
    <bean id="sayImpl" class="aop.demo.SayImpl"></bean>
    <!-- 聲明用于增強的攔截器對象 -->
    <bean id="sayImplAroundAdvice" class="aop.demo.SayImplAroundAdvice"></bean>
    <!-- 配置一個切面 -->
    <bean id="sayAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
        <property name="advice" ref="sayImplAroundAdvice"/>            <!-- 增強 -->
        <property name="pattern" value="aop.demo.SayImpl.s.*"/> <!-- 切點(正則表達式) -->
    </bean>   
   
    <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator">
        <property name="optimize" value="true"/>
    </bean>
</beans>

在此增加一個DefaultAdvisorAutoProxyCreator,原先的代理就不需要啦。然后再看一下客戶端調(diào)用直接改成調(diào)用SayImpl,看看能不能實現(xiàn)代理:

package aop.demo;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Client {
    @SuppressWarnings("resource")
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("aop/demo/spring.xml");
        ISay say = (ISay)context.getBean("sayImpl");
        say.say();
        say.noaop();
    }
}

輸出:

    大家好:我是5207.希望大家多多點贊.
    別aop我

效果達成,只不過這里的關鍵是DefaultAdvisorAutoProxyCreater是怎么做的呢?看了看代碼發(fā)現(xiàn)其主要是借助Ioc容器在初始化對象時完成的代理的自動生成的。在BeanPostProccer的postProcessAfterInitialization過程中完成了對代理的生成。具體的原理可以參考引用中的文章,太累了不寫了。

Release Notes

Popular Entries