1. 程式人生 > >使用CGLIB實現AOP功能與AOP概念解釋

使用CGLIB實現AOP功能與AOP概念解釋

spa 責任 構建 tca ebe invoke urn img use

使用CGLIB實現AOP功能

在Java裏面,我們要產生某個對象的代理對象,這個對象必須要有一個特點,即這個對象必須實現一個接口,動態代理技術只能基於接口進行代理。有時候我們在做開發的時候,這個對象就沒有實現接口,有人可能會說,它既然沒有接口,那我就給它定義一個接口,這是不行的,因為有時候我們拿到一個對象,而這個對象是服務器產生給我們的,是服務器提供給我們的,又不是我們自己寫的,動不動就給它定義一個接口,給它找個爸爸,哪那行呢?但我們現在要對它進行增強,這時用動態代理技術就不行了,動態代理技術只能是基於接口,那如果這個對象沒有接口,又該怎麽做呢?
那這時我們就需要使用另外一套API——CGLIB了,這套API,即使沒有接口,它也可以幫我們產生這個對象的代理對象。它的內部是怎麽去產生這個對象的代理對象的呢?——實際上產生的是這個對象的子類,也即我們把一個對象交給CGLIB,它返回出來的似乎是一個代理對象(但它不是要產生一個對象的代理對象),但其實這個代理對象就是這個對象的子類,利用子類的方式來創建代理對象。在Spring裏面就是這樣做的,Spring裏面有一個AOP編程(即面向切面編程,說白了就是動態代理,我們經常會交給Spring一個對象,它就會返回代理對象給我們,它在返回代理對象的時候,首先會檢查我們這個對象有沒有實現一個接口,如果我們這個類有接口,它使用Java的動態代理技術來幫我們構建出代理對象;如果我們這個類沒有實現接口,它會使用CGLIB這套API,采用創建子類的方式來創建代理對象)


本文是建立在使用JDK中的Proxy技術實現AOP功能的案例的基礎之上的,若要使用CGLIB這套API實現AOP功能,就要將其所需要的jar包導入項目中,所需的jar包有:

  • asm-2.2.3.jar
  • cglib-nodep-2.2.jar

首先將PersonServiceBean類的代碼修改為:

public class PersonServiceBean {
    private String user = null;

    public String getUser() {
        return user;
    }

    public PersonServiceBean() {}

    public PersonServiceBean(String user) {
        this.user = user;
    }

    public void save(String name) {
        System.out.println("我是save()方法");
    }

    public void update(String name, Integer personid) {
        System.out.println("我是update()方法");
    }

    public String getPersonName(Integer personid) {
        System.out.println("我是getPersonName()方法");
        return "xxx";
    }
}

可發現PersonServiceBean類沒實現一個接口,現在要想產生PersonServiceBean類的代理對象,這時就不能不使用CGLIB這套API了。
我們在it.cast.aop包下新建一個類——CGlibProxyFactory.java,與JDKProxyFactory類相似,都用於創建代理對象,其代碼為:

public class CGlibProxyFactory implements MethodInterceptor {
    private Object targetObject; // 代理的目標對象

    public Object createProxyInstance(Object targetObject) {
        this.targetObject = targetObject; 

        Enhancer enhancer = new Enhancer(); // 該類用於生成代理對象
        enhancer.setSuperclass(this.targetObject.getClass()); // 設置目標類為代理對象的父類
        enhancer.setCallback(this); // 設置回調用對象為本身

        return enhancer.create();
    }

    @Override
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        PersonServiceBean bean = (PersonServiceBean)this.targetObject;
        Object result = null; 
        if (bean.getUser() != null) { // 有權限
            result = methodProxy.invoke(targetObject, args); // 把方法調用委派給目標對象
        }
        return result;
    }
}
  • 1

結論:CGLIB可以生成目標類的子類,並重寫父類非final修飾符的方法
接著就要修改AOPTest類的代碼了,我們先將AOPTest類的代碼改為:

public class AOPTest {

    @Test
    public void proxyTest() {
        JDKProxyFactory factory = new JDKProxyFactory();
        PersonService service = (PersonService) factory.createProxyInstance(new PersonServiceBean());
        service.save("888");
    }

    @Test
    public void proxyTest2() {
        CGlibProxyFactory factory = new CGlibProxyFactory();
        PersonServiceBean service = (PersonServiceBean) factory.createProxyInstance(new PersonServiceBean("xxx"));
        service.save("999");
    }

}
  • 1
  • 2

測試proxyTest2()方法,Eclipse控制臺會打印:
技術分享
若是將AOPTest類的代碼改為:

public class AOPTest {

    @Test
    public void proxyTest() {
        JDKProxyFactory factory = new JDKProxyFactory();
        PersonService service = (PersonService) factory.createProxyInstance(new PersonServiceBean());
        service.save("888");
    }

    @Test
    public void proxyTest2() {
        CGlibProxyFactory factory = new CGlibProxyFactory();
        PersonServiceBean service = (PersonServiceBean) factory.createProxyInstance(new PersonServiceBean());
        service.save("999");
    }

}
  • 1

再次測試proxyTest2()方法,Eclipse控制臺什麽都不會打印。
如要查看源碼,可點擊使用JDK中的Proxy技術實現AOP功能與使用CGLIB實現AOP功能進行下載。

AOP概念解釋

AOP用在哪些方面:AOP能夠將那些與業務無關,卻為業務模塊所共同調用的邏輯或責任,例如事務處理、日誌管理、權限控制,異常處理等,封裝起來,便於減少系統的重復代碼,降低模塊間的耦合度,並有利於未來的可操作性和可維護性。

AOP中的概念

Aspect(切面):指橫切性關註點的抽象即為切面,它與類相似,只是兩者的關註點不一樣,類是對物體特征的抽象,而切面是橫切性關註點的抽象。
joinpoint(連接點):所謂連接點是指那些被攔截到的點(可以是方法、屬性、或者類的初始化時機(可以是Action層、Service層、dao層))。在Spring中,這些點指的是方法,因為Spring只支持方法類型的連接點,實際上joinpoint還可以是field或類構造器。
Pointcut(切入點):所謂切入點是指我們要對那些joinpoint進行攔截的定義,也即joinpoint的集合。
Advice(通知):所謂通知是指攔截到joinpoint之後所要做的事情就是通知。通知分為前置通知、後置通知、異常通知、最終通知、環繞通知。我們就以CGlibProxyFactory類的代碼為例進行說明:

public class CGlibProxyFactory implements MethodInterceptor {
    private Object targetObject; // 代理的目標對象

    public Object createProxyInstance(Object targetObject) {
        this.targetObject = targetObject; 

        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(this.targetObject.getClass()); // 設置目標類為代理對象的父類
        enhancer.setCallback(this);

        return enhancer.create();
    }

    // 從另一種角度看: 整個方法可看作環繞通知
    @Override
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        PersonServiceBean bean = (PersonServiceBean)this.targetObject;
        Object result = null; 
        if (bean.getUser() != null) { // 有權限
            // ...... advice() ----> 前置通知(所謂通知,就是我們攔截到業務方法之後所要幹的事情)
            try {
                result = methodProxy.invoke(targetObject, args); // 把方法調用委派給目標對象
                // ...... afteradvice() ----> 後置通知
            } catch (RuntimeException e) {
                // ...... exceptionadvice() ----> 異常通知
            } finally {
                // ...... finallyadvice() ----> 最終通知
            }
        }
        return result;
    }
}

Target(目標對象):代理的目標對象。
Weave(織入):指將aspects應用到target對象並導致proxy對象創建的過程稱為織入。
Introduction(引入):在不修改類代碼的前提下,Introduction可以在運行期為類(代理類)動態地添加一些方法或Field。
AOP帶來的好處:降低模塊的耦合度;使系統容易擴展;更好的代碼復用性

使用CGLIB實現AOP功能與AOP概念解釋