中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java反射與動態代理的介紹

發布時間:2021-08-30 16:47:43 來源:億速云 閱讀:149 作者:chen 欄目:開發技術

這篇文章主要講解了“Java反射與動態代理的介紹”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“Java反射與動態代理的介紹”吧!

目錄
  • 一、反射

  • 二、動態代理

    • 1、JDK代理

    • 2、CGLIB代理

    • 3、JDK代理與CGLIB代理對比


一、反射

概念:在運行狀態中,對于任意的一個類,都能夠知道這個類的所有字段和方法,對任意一個對象都能夠通過反射機制調用一個類的任意方法

實現方法:JVM在第一次加載某個類時會生成一個Class對象,里面記錄了這個類的信息

鏈接:類加載機制(留坑)

二、動態代理

動態代理的作用:在不改變原代碼的基礎上增加新的功能,如日志、權限檢驗等

反射在動態代理中的應用:由于知道原類的字段、方法等信息,才可以通過代理類執行被代理類的方法

動態代理的實現有兩種

1、JDK代理

實現方法:通過創建一個代理類,這個代理類繼承于一個Proxy類,Proxy類中有一個InvocationHandler接口,這個接口持有被代理對象和一個invoke()方法。創建好代理類對象后,對該對象調用的方法都會交由invoke方法處理。invoke方法接受3個參數:代理對象、方法、參數列表。重寫invoke方法便可以在原方法的基礎上添加其他邏輯

一個JDK代理的簡單實現:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
// 定義一個接口
interface MyInterface {
    void fun();
}
// 用一個類去實現這個接口
class Person implements MyInterface {
    @Override
    public void fun() {
        System.out.println("Person實現接口方法");
    }
}
// 用一個類實現InvocationHandler接口
class MyInvocationHandler<T> implements InvocationHandler {
    //InvocationHandler持有的被代理對象
    T target;
    public MyInvocationHandler(T target) {
        this.target = target;
    }
    /**
     * proxy:代表動態代理對象
     * method:代表正在執行的方法
     * args:代表調用目標方法時傳入的實參
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("代理執行" + method.getName() + "方法"); // 在原方法的基礎上添加其他邏輯
        Object result = method.invoke(target, args); // 通過invoke方法調用原方法
        return result;
    }
}
public class Main {
    public static void main(String[] args) {
        Person person = new Person();
        InvocationHandler invocationHandler = new MyInvocationHandler<>(person);
        MyInterface proxy = (MyInterface) Proxy.newProxyInstance(Person.class.getClassLoader(),
                Person.class.getInterfaces(), invocationHandler);
        proxy.fun();
    }
}

輸出結果:

代理執行fun方法
Person實現接口方法

缺陷:只能代理接口方法,因為JDK代理需要繼承一個Proxy類,又由于Java的單繼承機制,導致代理類無法繼承父類的函數,只能實現接口

2、CGLIB代理

原理與JDK代理類似,區別在于CGLIB代理創建的代理類直接繼承于被代理類,所以可以實現被代理類的方法而非僅僅接口方法

一個簡單的CGLIB代理實現:

public class Main {
    public static void main(String[] args) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(Car.class);
        enhancer.setCallback(new MethodInterceptor() {
            @Override
            public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy)
                    throws Throwable {
                System.out.println("before");
                Object res = methodProxy.invokeSuper(obj, args);
                System.out.println("after");
                return res;
            }
        });
        Car car = (Car) enhancer.create();
        car.print();
    }
}
class Car {
    void print() {
        System.out.println("執行原方法");
    }
}

由于CGLIB并非JDK自帶,所以需要通過Maven引入一個依賴

<dependency>
    <groupId>org.sonatype.sisu.inject</groupId>
    <artifactId>cglib</artifactId>
    <version>3.1.1</version>
</dependency>

輸出結果:

before
執行原方法
after

3、JDK代理與CGLIB代理對比

1、JDK代理只能實現接口方法,而CGLIB代理既可以實現接口方法也可以實現類中自帶的方法

2、性能上,在JDK1.8,CGLIB3.1.1的環境上,每次創建一個代理類并執行同樣的方法

當執行10000次,JDK代理用時85ms,而CGLIB代理用時190ms,明顯JDK代理性能更佳;

當執行1000000(一百萬)次時,兩種代理耗時幾乎相等;

當執行10000000次時,CGLIB代理已經優于JDK代理。

所以在執行次數少時,JDK代理性能更好;反之CGLIB代理性能更好(但是重復執行多于1000000次的任務幾乎沒有吧

感謝各位的閱讀,以上就是“Java反射與動態代理的介紹”的內容了,經過本文的學習后,相信大家對Java反射與動態代理的介紹這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

阿拉善盟| 洞头县| 襄城县| 甘孜| 米脂县| 阿瓦提县| 安乡县| 平原县| 辽源市| 饶阳县| 武鸣县| 石景山区| 谢通门县| 栾川县| 柯坪县| 沁水县| 永德县| 阳曲县| 永定县| 曲阜市| 宜阳县| 电白县| 石渠县| 东乡族自治县| 图片| 新巴尔虎右旗| 泽普县| 津市市| 嘉禾县| 上饶县| 恩施市| 林州市| 格尔木市| 兴安县| 泗阳县| 凤山市| 上思县| 阳曲县| 堆龙德庆县| 五莲县| 万宁市|