JDK動態代理和CGLib動態代理的對比


JDK動態代理:利用反射機制生成一個實現代理接口的匿名類,在調用具體方法前調用InvokeHandler來處理。
CGLib動態代理:利用ASM(開源的Java字節碼編輯庫,操作字節碼)開源包,將代理對象類的class文件加載進來,通過修改其字節碼生成子類來處理。

1. JDK動態代理

1.1 角色

  1. Interface:對於JDK Proxy,業務類是需要一個Interface的。
  2. Proxy:Proxy類是動態產生的,這個類在調用Proxy.newProxyInstance()方法之后,產生一個Proxy類的實力。實際上,這個Proxy類也是存在的,不僅僅是類的實例,這個Proxy類可以保存在硬盤上。
  3. Method:對於業務委托類的每個方法,現在Proxy類里面都不用靜態顯示出來。
  4. InvocationHandler:這個類在業務委托類執行時,會先調用invoke方法。invoke方法在執行想要的代理操作,可以實現對業務方法的再包裝。

1.2 總結

  • JDK動態代理類實現了InvocationHandler接口,重寫的invoke方法。
  • JDK動態代理的基礎是反射機制(method.invoke(對象,參數))Proxy.newProxyInstance()。

1.3 動態代理步驟

  1. 創建被代理類及接口(JDK代理是接口代理)
  2. 創建Handle類實現 InvocationHandler接口 ,重寫invoke方法
  3. 通過Proxy的newProxyInstance()方法獲取代理類對象
  4. 通過代理類對象調用被代理類的方法

1.4 代碼實現

//接口類
public interface FoodService {
    public void makeNoodle();
    public void makeChicken();
}

//代理類,實現定義的接口
public class FoodServiceImpl implements FoodService {
    @Override
    public void makeNoodle() {
        System.out.println("make noodle");
    }

    @Override
    public void makeChicken() {
        System.out.println("make Chicken");
    }
}
public class JDKProxyFactory implements InvocationHandler {
    private Object target;

    public JDKProxyFactory(Object target) {
        super();
        this.target = target;
    }

    // 創建代理對象
    public Object createProxy() {
        // 1.得到目標對象的類加載器
        ClassLoader classLoader = target.getClass().getClassLoader();
        // 2.得到目標對象的實現接口
        Class<?>[] interfaces = target.getClass().getInterfaces();
        // 3.第三個參數需要一個實現invocationHandler接口的對象
        Object newProxyInstance = Proxy.newProxyInstance(classLoader, interfaces, this);
        return newProxyInstance;
    }


    // 第一個參數:代理對象.一般不使用;第二個參數:需要增強的方法;第三個參數:方法中的參數
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("這是增強方法前......");
        Object invoke = method.invoke(target, args);
        System.out.println("這是增強方法后......");
        return invoke;
    }

    public static void main(String[] args) {
        // 1.創建對象
        FoodServiceImpl foodService = new FoodServiceImpl();
        // 2.創建代理對象
        JDKProxyFactory proxy = new JDKProxyFactory(foodService);
        // 3.調用代理對象的增強方法,得到增強后的對象
        FoodService createProxy = (FoodService) proxy.createProxy();
        createProxy.makeChicken();
    }

}

2. CGLib動態代理

強制使用CGLib

<!-- proxy-target-class="false"默認使用JDK動態代理 -->
<aop:aspectj-autoproxy proxy-target-class="true"/>

<aop-config proxy-target-class="true">
<!-- 切面詳細配置 -->
</aop-config>
public class CglibProxyFactory implements MethodInterceptor {
    //得到目標對象
    private Object target;

    //使用構造方法傳遞目標對象
    public CglibProxyFactory(Object target) {
        super();
        this.target = target;
    }

    //創建代理對象
    public Object createProxy(){
        //1.創建Enhancer
        Enhancer enhancer = new Enhancer();
        //2.傳遞目標對象的class
        enhancer.setSuperclass(target.getClass());
        //3.設置回調操作
        enhancer.setCallback(this);

        return enhancer.create();
    }


    //參數一:代理對象;參數二:需要增強的方法;參數三:需要增強方法的參數;參數四:需要增強的方法的代理
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        System.out.println("這是增強方法前......");
        Object invoke = methodProxy.invoke(target, args);
        System.out.println("這是增強方法后......");
        return invoke;
    }

    public static void main(String[] args) {
        // 1.創建對象
        FoodServiceImpl foodService = new FoodServiceImpl();
        // 2.創建代理對象
        CglibProxyFactory proxy = new CglibProxyFactory(foodService);
        // 3.調用代理對象的增強方法,得到增強后的對象
        FoodService createProxy = (FoodService) proxy.createProxy();
        createProxy.makeChicken();
    }
}

3. 兩者區別

  • JDK代理只能對實現接口的類生成代理;CGLib是針對類實現代理,對指定的類生成一個子類,並覆蓋其中的方法,這種通過繼承類的實現方式,不能代理final修飾的類。
  • JDK代理使用的是反射機制實現aop的動態代理,CGLib代理使用字節碼處理框架ASM,通過修改字節碼生成子類。所以jdk動態代理的方式創建代理對象效率較高,執行效率較低,CGLib創建效率較低,執行效率高。
  • JDK動態代理機制是委托機制,具體說動態實現接口類,在動態生成的實現類里面委托hanlder去調用原始實現類方法,CGLib則使用的繼承機制,具體說被代理類和代理類是繼承關系,所以代理類是可以賦值給被代理類的,如果被代理類有接口,那么代理類也可以賦值給接口。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM