首頁 >Java >java教程 >動態代理的兩種方式是什麼

動態代理的兩種方式是什麼

青灯夜游
青灯夜游原創
2022-01-06 17:32:0519238瀏覽

兩種方式為:1、JDK動態代理,利用反射機制產生一個實作代理介面的匿名類,在呼叫具體方法前呼叫InvokeHandler來處理;2、CGLIB動態代理,利用asm開源包,對代理物件類別的class檔案載入進來,透過修改其字節碼產生子類別來處理。

動態代理的兩種方式是什麼

本教學操作環境:windows7系統、java8版、DELL G3電腦。

動態代理程式是反射的一個非常重要的應用場景。動態代理常被用於一些 Java 框架。例如 Spring 的 AOP ,Dubbo 的 SPI 接口,就是基於 Java 動態代理實現的。

動態代理程式的方式有兩種:

  • #JDK動態代理程式:利用反射機制產生實作代理接口的匿名類,在呼叫具體方法前呼叫InvokeHandler來處理。

  • CGLIB動態代理程式:利用ASM(開源的Java字節碼編輯庫,操作字節碼)開源包,將代理物件類別的class檔案載入進來,透過修改其字節碼產生子類別來處理。

區別:JDK代理只能對實作介面的類別產生代理;CGlib是針對類別實作代理,對指定的類別產生子類,並且覆寫其中的方法,這種透過繼承類別的實作方式,不能代理final修飾的類別。

強制使用CGlib

<!-- proxy-target-class="false"默认使用JDK动态代理 -->
<aop:aspectj-autoproxy proxy-target-class="true"/>

<aop-config proxy-target-class="true">
<!-- 切面详细配置 -->
</aop-config>

具體程式碼範例:

/**
 * 目标接口类
 */
public interface UserManager {    
    public void addUser(String id, String password);    
    public void delUser(String id);    
}
/**
 * 接口实现类
 */
public class UserManagerImpl implements UserManager {    
    
    @Override
    public void addUser(String id, String password) {    
        System.out.println("调用了UserManagerImpl.addUser()方法!");
    }    
    
    @Override
    public void delUser(String id) {    
        System.out.println("调用了UserManagerImpl.delUser()方法!");
    }    
}
/**
 * JDK动态代理类
 */
public class JDKProxy implements InvocationHandler {    
    
    // 需要代理的目标对象
    private Object targetObject;    
    
    public Object newProxy(Object targetObject) {
        // 将目标对象传入进行代理    
        this.targetObject = targetObject;
        // 返回代理对象 
        return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);
    }    
    
    // invoke方法
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 进行逻辑处理的函数
        checkPopedom();
        Object ret = null;
        // 调用invoke方法
        ret = method.invoke(targetObject, args);
        return ret;
    }    
    
    private void checkPopedom() {
        // 模拟检查权限   
        System.out.println("检查权限:checkPopedom()!");    
    }    
}
/**
 * CGlib动态代理类
 */
 public class CGLibProxy implements MethodInterceptor {    
    
    // CGlib需要代理的目标对象
    private Object targetObject;
    
    public Object createProxyObject(Object obj) {
        this.targetObject = obj;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(obj.getClass());
        enhancer.setCallback(this);
        Object proxyObj = enhancer.create();
        return proxyObj;
    }
    
    @Override
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        Object obj = null;
        // 过滤方法
        if ("addUser".equals(method.getName())) {
            // 检查权限
            checkPopedom();
        }
        obj = method.invoke(targetObject, args);
        return obj;
    }    
    
    private void checkPopedom() {
        System.out.println("检查权限:checkPopedom()!");
    }
}
/**
 * 测试类
 */
public class ProxyTest {
    
    public static void main(String[] args) {
        UserManager userManager = (UserManager)new CGLibProxy().createProxyObject(new UserManagerImpl());
        System.out.println("CGLibProxy:");
        userManager.addUser("tom", "root");
        System.out.println("JDKProxy:");
        JDKProxy jdkProxy = new JDKProxy();
        UserManager userManagerJDK = (UserManager)jdkProxy.newProxy(new UserManagerImpl());
        userManagerJDK.addUser("tom", "root");
    }
}
// 运行结果
CGLibProxy:
检查权限checkPopedom()!
调用了UserManagerImpl.addUser()方法!
JDKProxy:
检查权限checkPopedom()!
掉用了UserManagerImpl.addUser()方法!

總結:

1、JDK代理程式使用的是反射機制實作aop的動態代理,CGLIB代理使用字節碼處理框架asm,透過修改字節碼來產生子類別。所以jdk動態代理的方式創建代理對象效率較高,執行效率較低,cglib創建效率較低,執行效率高;

2、JDK動態代理機制是委託機制,具體說動態實現接口類,在動態產生的實作類別裡面委託hanlder去呼叫原始實作類別方法,CGLIB則使用的繼承機制,具體說被代理類別和代理類別是繼承關係,所以代理類別是可以賦值給被代理類別的,如果被代理類別有接口,那麼代理類別也可以賦值給接口。

(推薦教學:java入門教學

以上是動態代理的兩種方式是什麼的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn