成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

面試官:為什么JDK動態代理只能代理接口?

開發 前端
我們使用CGLIB的Enhancer?類和MethodInterceptor?接口來創建代理對象。RealObject?類不再需要實現接口,而是直接作為代理對象的類型。在CGLIBProxy?類中,我們實現了MethodInterceptor?接口,并在intercept()方法中添加了額外的邏輯。

哈嘍,大家好,我是了不起。如題,這個問題應該面試常考題,當你遇到這個問題時,如果你能回答出來JDK動態代理的原理,然后引申Cglib 動態代理,那么這個面試官一定會對你刮目相看。

圖片

在Java中,動態代理是一種機制,允許在運行時動態地創建代理對象來代替某個實際對象,從而在其前后執行額外的邏輯。

為什么JDK動態代理只能代理接口實現類,原因是JDK動態代理是基于接口實現的。

當你使用Proxy類創建代理對象時,你需要指定一個接口列表來表示代理對象所應該實現的接口,這些接口就成為代理對象的類型。

具體來說,代理對象的方法調用會被轉發到實現InvocationHandler接口的類中的invoke()方法。這個invoke()方法接受三個參數:代理對象本身、被調用的方法對象和方法的參數數組。invoke()方法需要返回被代理方法調用的結果。

由于代理對象的類型是由接口列表決定的,因此只有實現了接口的類才能被代理。如果你想代理一個類而不是一個接口,你需要使用其他的代理技術,比如CGLIB。

1、JDK動態代理代碼實例

下面是一個簡單的示例代碼,展示了如何使用JDK動態代理來創建代理對象。

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class ProxyDemo {
    public static void main(String[] args) {
        RealObject real = new RealObject();
        InvocationHandler handler = new DynamicProxy(real);

        // 創建代理對象
        MyInterface proxy = (MyInterface) Proxy.newProxyInstance(
                MyInterface.class.getClassLoader(),
                new Class<?>[] { MyInterface.class },
                handler);

        // 調用代理對象的方法
        proxy.doSomething();
    }
}

interface MyInterface {
    void doSomething();
}

class RealObject implements MyInterface {
    public void doSomething() {
        System.out.println("RealObject doSomething");
    }
}

class DynamicProxy implements InvocationHandler {
    private Object target;

    public DynamicProxy(Object target) {
        this.target = target;
    }

    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        System.out.println("Before method invocation");
        Object result = method.invoke(target, args);
        System.out.println("After method invocation");
        return result;
    }
}

在上面的代碼中,RealObject實現了MyInterface接口,它是我們要代理的實際對象。DynamicProxy類實現了InvocationHandler接口,并在invoke()方法中添加了額外的邏輯,用于在代理對象方法調用前后執行。

在main()方法中,我們使用Proxy.newProxyInstance()方法創建代理對象。我們指定了MyInterface接口作為代理對象類型,并將DynamicProxy對象作為代理對象的InvocationHandler。

最后,我們調用代理對象的doSomething()方法,并觀察控制臺輸出的結果。

需要注意的是,代理對象的方法調用都會被轉發到DynamicProxy類的invoke()方法中進行處理,因此在這個示例中,實際的RealObject對象的doSomething()方法的執行是在invoke()方法中通過反射進行的。

總結一下,JDK動態代理只能代理接口實現類,原因是JDK動態代理是基于接口實現的,代理對象的類型由接口列表決定。如果你想代理一個類而不是一個接口,你需要使用其他的代理技術,比如CGLIB。

2、Cglib 代碼演示

以下是CGLIB代理的示例代碼。

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

public class CGLIBProxyDemo {
    public static void main(String[] args) {
        RealObject real = new RealObject();
        MethodInterceptor handler = new CGLIBProxy(real);

        // 創建代理對象
        RealObject proxy = (RealObject) Enhancer.create(
                RealObject.class,
                handler);

        // 調用代理對象的方法
        proxy.doSomething();
    }
}

class CGLIBProxy implements MethodInterceptor {
    private Object target;

    public CGLIBProxy(Object target) {
        this.target = target;
    }

    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("Before method invocation");
        Object result = proxy.invoke(target, args);
        System.out.println("After method invocation");
        return result;
    }
}

在上面的示例中,我們使用CGLIB的Enhancer類和MethodInterceptor接口來創建代理對象。RealObject類不再需要實現接口,而是直接作為代理對象的類型。在CGLIBProxy類中,我們實現了MethodInterceptor接口,并在intercept()方法中添加了額外的邏輯。

在main()方法中,我們使用Enhancer.create()方法創建代理對象。我們指定了RealObject類作為代理對象類型,并將CGLIBProxy對象作為代理對象的MethodInterceptor。最后,我們調用代理對象的doSomething()方法,并觀察控制臺輸出的結果。

需要注意的是,CGLIB代理使用字節碼技術來生成代理對象,因此它的效率比JDK動態代理要高,但是它也需要額外的庫依賴。

3、兩者優缺點

JDK動態代理和CGLIB代理都有它們自己的優缺點。

JDK動態代理的優點:

  • JDK動態代理是Java標準庫的一部分,因此它不需要引入任何外部依賴。
  • JDK動態代理只需要實現接口即可生成代理對象,不需要改變原有類的結構。
  • 由于JDK動態代理是基于接口實現的,因此它更適合用于代理接口實現類的場景。

JDK動態代理的缺點:

  • JDK動態代理只能代理實現了接口的類,無法代理沒有實現接口的類。
  • JDK動態代理在生成代理對象時,需要使用反射機制,因此它的效率相對較低。

CGLIB代理的優點:

  • CGLIB代理是基于字節碼技術實現的,因此它的效率比JDK動態代理更高。
  • CGLIB代理可以代理沒有實現接口的類。

CGLIB代理的缺點:

  • CGLIB代理需要引入外部依賴。
  • CGLIB代理在生成代理對象時,需要改變原有類的結構,因此它可能會引起一些問題,例如無法代理final類或final方法等問題。

綜上所述,JDK動態代理適用于代理接口實現類的場景,而CGLIB代理適用于代理沒有實現接口的類的場景。如果你需要代理接口實現類,而且不想引入額外的依賴,那么JDK動態代理是一個不錯的選擇;如果你需要代理沒有實現接口的類,那么CGLIB代理可能更適合你的需求。

責任編輯:武曉燕 來源: Java技術指北
相關推薦

2020-12-29 05:34:00

動態代理

2021-12-08 06:53:29

面試動態代理

2022-02-22 22:44:46

接口源碼對象

2023-12-06 08:23:44

代理模式設計模式

2022-04-29 08:17:38

RPC遠程代理代理模式

2015-09-22 11:09:47

Java 8動態代理

2021-04-22 09:58:15

JDK代理動態

2022-07-06 13:48:24

RedisSentinel機制

2021-03-16 21:42:37

反向代理正向代理

2023-12-06 09:10:28

JWT微服務

2020-10-24 15:50:54

Java值傳遞代碼

2021-02-19 10:02:57

HTTPSJava安全

2024-02-04 16:51:47

2021-01-21 07:53:29

面試官Promis打印e

2021-11-05 07:47:56

代理模式對象

2022-09-01 10:40:29

SpringAOPJDK

2024-04-15 10:16:38

2021-12-20 10:30:33

forforEach前端

2023-12-20 14:35:37

Java虛擬線程

2011-04-06 11:41:25

Java動態代理
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 在线观看亚洲专区 | 国产乱码精品1区2区3区 | 精品成人| 成人在线观 | 精品99在线 | 欧美在线视频网 | 日本免费一区二区三区四区 | 成人午夜黄色 | 中文一区二区视频 | 欧美美女爱爱 | 国产在线中文字幕 | 中文字幕一区二区三区四区五区 | 成人三区 | 国产91网站在线观看 | 日韩成人免费视频 | 欧美精品一区二区免费视频 | 久草网站 | 激情视频中文字幕 | 国产午夜精品视频 | 一区二区三区四区av | 国产精品高潮呻吟久久av黑人 | 波多野结衣一区二区三区 | 日产久久| 黄色一级网 | 美女福利视频一区 | 久久国产精品视频免费看 | 国产91综合 | 亚洲国产一区在线 | 夜夜爽99久久国产综合精品女不卡 | 一区二区三区av | 日韩欧美三区 | 国产美女精品视频 | 色www精品视频在线观看 | 亚洲一区二区三 | 欧美日韩成人影院 | 精品国产伦一区二区三区观看体验 | 国产精品18毛片一区二区 | 国产在线不卡 | 国产精品亚洲成在人线 | 国产精品国产a | 欧美三级在线 |