欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文搞懂Java常見的三種代理模式(靜態(tài)代理、動(dòng)態(tài)代理和cglib代理)

 更新時(shí)間:2023年08月09日 10:35:25   作者:半畝方塘立身  
Java中常見的三種代理模式是靜態(tài)代理模式、動(dòng)態(tài)代理模式和CGLIB代理模式,本文就來給大家詳細(xì)的講解一下這三種代理模式,感興趣的小伙伴跟著小編一起來看看吧

Java靜態(tài)代理

Java中的靜態(tài)代理是一種設(shè)計(jì)模式,它通過創(chuàng)建一個(gè)代理類來代替原始類,從而控制對(duì)原始類的訪問。代理類和原始類都實(shí)現(xiàn)相同的接口,使得客戶端在使用時(shí)無需關(guān)心具體的實(shí)現(xiàn)細(xì)節(jié)。靜態(tài)代理在編譯時(shí)就已經(jīng)確定代理類和原始類的關(guān)系,因此稱為靜態(tài)代理。

下面是Java靜態(tài)代理UML類圖:

image.png

示例代碼: 假設(shè)有一個(gè)文件上傳的接口 FileUploader ,其真實(shí)實(shí)現(xiàn)為 RealFileUploader ,我們需要在上傳文件前后記錄日志,這時(shí)就可以使用靜態(tài)代理。

首先,定義抽象接口 FileUploader

public interface FileUploader {
    void upload(String file);
}

然后,實(shí)現(xiàn)真實(shí)上傳 RealFileUploader

public class RealFileUploader implements FileUploader {
    @Override
    public void upload(String file) {
        // 實(shí)際的文件上傳邏輯
        System.out.println("Uploading file: " + file);
    }
}

接下來,實(shí)現(xiàn)代理 ProxyFileUploader

public class ProxyFileUploader implements FileUploader {
    private FileUploader realFileUploader;
    public ProxyFileUploader(FileUploader realFileUploader) {
        this.realFileUploader = realFileUploader;
    }
    @Override
    public void upload(String file) {
        // 額外的處理,比如記錄日志
        System.out.println("Before uploading: " + file);
        // 調(diào)用真實(shí)主題的上傳方法
        realFileUploader.upload(file);
        // 額外的處理,比如記錄日志
        System.out.println("After uploading: " + file);
    }
}

在使用時(shí),我們可以這樣創(chuàng)建代理類并進(jìn)行文件上傳:

public class Main {
    public static void main(String[] args) {
        // 創(chuàng)建真實(shí)主題
        FileUploader realFileUploader = new RealFileUploader();
        // 創(chuàng)建代理主題,并將真實(shí)主題傳入
        FileUploader proxyFileUploader = new ProxyFileUploader(realFileUploader);
        // 通過代理主題進(jìn)行文件上傳
        proxyFileUploader.upload("example.txt");
    }
}

運(yùn)行上述代碼,輸出將會(huì)是:

Before uploading: example.txt
Uploading file: example.txt
After uploading: example.txt

Java動(dòng)態(tài)代理

Java動(dòng)態(tài)代理是一種在運(yùn)行時(shí)創(chuàng)建代理類的機(jī)制,它允許在不提前知道代理類的具體類型的情況下,動(dòng)態(tài)地創(chuàng)建一個(gè)代理對(duì)象來代替原始類。相比于靜態(tài)代理,動(dòng)態(tài)代理更加靈活,可以代理任意的接口類型,不需要為每個(gè)被代理的類編寫專門的代理類,而是通過Java的反射機(jī)制在運(yùn)行時(shí)動(dòng)態(tài)生成代理類。動(dòng)態(tài)代理主要使用 java.lang.reflect.Proxy 類和 java.lang.reflect.InvocationHandler 接口來實(shí)現(xiàn)。動(dòng)態(tài)代理又被稱為JDK代理或接口代理。

下面看下動(dòng)態(tài)代理的UML圖:

java動(dòng)態(tài)代理 (1).png

示例代碼: 與前面的靜態(tài)代理示例相同,我們?nèi)匀皇褂梦募蟼鞯慕涌?FileUploader ,其真實(shí)實(shí)現(xiàn)為 RealFileUploader 。接下來,我們將通過動(dòng)態(tài)代理在上傳文件前后記錄日志。

首先,定義抽象主題接口 FileUploader 和實(shí)現(xiàn)真實(shí)主題 RealFileUploader ,與之前相同,不再重復(fù)。

然后,實(shí)現(xiàn) InvocationHandler 接口,創(chuàng)建動(dòng)態(tài)代理的處理器類 DynamicProxyHandler

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class DynamicProxyHandler implements InvocationHandler {
    private Object realObject;
    public DynamicProxyHandler(Object realObject) {
        this.realObject = realObject;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 額外的處理,比如記錄日志
        System.out.println("Before invoking: " + method.getName());
        // 調(diào)用真實(shí)主題的對(duì)應(yīng)方法
        Object result = method.invoke(realObject, args);
        // 額外的處理,比如記錄日志
        System.out.println("After invoking: " + method.getName());
        return result;
    }
}

在使用時(shí),我們可以這樣創(chuàng)建動(dòng)態(tài)代理對(duì)象并進(jìn)行文件上傳:

import java.lang.reflect.Proxy;
public class Main {
    public static void main(String[] args) {
        // 創(chuàng)建真實(shí)主題
        FileUploader realFileUploader = new RealFileUploader();
        // 創(chuàng)建動(dòng)態(tài)代理的處理器
        InvocationHandler handler = new DynamicProxyHandler(realFileUploader);
        // 創(chuàng)建代理類
        FileUploader proxyFileUploader = (FileUploader) Proxy.newProxyInstance(
                FileUploader.class.getClassLoader(),
                new Class[]{FileUploader.class},
                handler
        );
        // 通過動(dòng)態(tài)代理進(jìn)行文件上傳
        proxyFileUploader.upload("example.txt");
    }
}

運(yùn)行上述代碼,輸出將會(huì)是:

Before invoking: upload
Uploading file: example.txt
After invoking: upload

同樣,通過動(dòng)態(tài)代理,我們?cè)谖募蟼髑昂蟪晒μ砑恿祟~外的處理(記錄日志),同時(shí)客戶端代碼無需關(guān)心具體的日志記錄邏輯,實(shí)現(xiàn)了解耦。動(dòng)態(tài)代理在很多場(chǎng)景下非常有用,例如AOP(面向切面編程)等。

CGLIB代理

CGLIB(Code Generation Library)是一個(gè)開源的第三方庫,用于在Java運(yùn)行時(shí)生成字節(jié)碼并創(chuàng)建代理類。與Java標(biāo)準(zhǔn)庫中的動(dòng)態(tài)代理(基于接口)不同,CGLIB代理可以代理普通類,即使它們沒有實(shí)現(xiàn)任何接口。CGLIB使用ASM庫來生成字節(jié)碼,并通過繼承的方式創(chuàng)建代理類,因此也被稱為子類代理。CGLIB廣泛用于各種框架和庫中,如Spring AOP。

下面詳細(xì)介紹CGLIB代理的結(jié)構(gòu)和示例:

  1. 示例代碼: 假設(shè)有一個(gè)簡單的類 Calculator ,我們希望在其方法執(zhí)行前后記錄日志,可以使用CGLIB代理。

首先,引入CGLIB庫,可以通過Maven添加依賴:

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>版本號(hào)</version>
</dependency>

然后,定義普通類 Calculator

public class Calculator {
    public int add(int a, int b) {
        return a + b;
    }
}

接下來,創(chuàng)建一個(gè)代理類生成器 CalculatorProxyGenerator

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 CalculatorProxyGenerator implements MethodInterceptor {
    private Object target;
    public CalculatorProxyGenerator(Object target) {
        this.target = target;
    }
    public Object createProxy() {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(target.getClass());
        enhancer.setCallback(this);
        return enhancer.create();
    }
    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("Before invoking: " + method.getName());
        Object result = proxy.invokeSuper(obj, args);
        System.out.println("After invoking: " + method.getName());
        return result;
    }
}

intercept 方法中,我們執(zhí)行了代理類的方法(通過 proxy.invokeSuper(obj, args) ),在方法執(zhí)行前后添加了額外的處理。

最后,在使用時(shí),我們可以這樣創(chuàng)建代理對(duì)象并調(diào)用方法:

public class Main {
    public static void main(String[] args) {
        Calculator calculator = new Calculator();
        CalculatorProxyGenerator proxyGenerator = new CalculatorProxyGenerator(calculator);
        Calculator proxyCalculator = (Calculator) proxyGenerator.createProxy();
        int result = proxyCalculator.add(3, 5);
        System.out.println("Result: " + result);
    }
}

運(yùn)行上述代碼,輸出將會(huì)是:

Before invoking: add
After invoking: add
Result: 8

這樣,通過CGLIB代理,我們成功在方法執(zhí)行前后添加了額外的處理(記錄日志),實(shí)現(xiàn)了解耦。CGLIB代理在不需要接口的情況下也能很好地完成代理任務(wù),但由于它是通過繼承的方式生成代理類,可能會(huì)影響某些場(chǎng)景,比如無法代理 final 方法。

動(dòng)態(tài)代理與CGLIB代理的區(qū)別

Spring在5.X之前默認(rèn)的動(dòng)態(tài)代理實(shí)現(xiàn)一直是jdk動(dòng)態(tài)代理。但是從5.X開始,spring就開始默認(rèn)使用Cglib來作為動(dòng)態(tài)代理實(shí)現(xiàn)。并且springboot從2.X開始也轉(zhuǎn)向了Cglib動(dòng)態(tài)代理實(shí)現(xiàn)。

為什么spring體系整體轉(zhuǎn)投Cglib呢,jdk動(dòng)態(tài)代理又有什么缺點(diǎn)呢?

  • jdk動(dòng)態(tài)代理只能基于接口,代理生成的對(duì)象只能賦值給接口變量,而Cglib就不存在這個(gè)問題,Cglib是通過生成子類來實(shí)現(xiàn)的,代理對(duì)象既可以賦值給實(shí)現(xiàn)類,又可以賦值給接口。
  • Cglib速度比jdk動(dòng)態(tài)代理更快,性能更好。

以上就是一文搞懂Java的三種代理模式(靜態(tài)代理、動(dòng)態(tài)代理和cglib代理)的詳細(xì)內(nèi)容,更多關(guān)于Java代理模式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot項(xiàng)目中的多數(shù)據(jù)源支持的方法

    SpringBoot項(xiàng)目中的多數(shù)據(jù)源支持的方法

    本篇文章主要介紹了SpringBoot項(xiàng)目中的多數(shù)據(jù)源支持的方法,主要介紹在SpringBoot項(xiàng)目中利用SpringDataJpa技術(shù)如何支持多個(gè)數(shù)據(jù)庫的數(shù)據(jù)源,有興趣的可以了解一下
    2017-10-10
  • Java設(shè)計(jì)模式中觀察者模式詳解

    Java設(shè)計(jì)模式中觀察者模式詳解

    觀察者模式是極其重要的一個(gè)設(shè)計(jì)模式,也是我?guī)啄觊_發(fā)過程中使用最多的設(shè)計(jì)模式,本文首先概述觀察者模式的基本概念和Demo實(shí)現(xiàn),接著是觀察者模式在Java和Spring中的應(yīng)用,最后是對(duì)觀察者模式的應(yīng)用場(chǎng)景和優(yōu)缺點(diǎn)進(jìn)行總結(jié)
    2022-11-11
  • mybatis自定義類型處理器的實(shí)現(xiàn)

    mybatis自定義類型處理器的實(shí)現(xiàn)

    在MyBatis使用中,有時(shí)需要對(duì)特定數(shù)據(jù)類型進(jìn)行定制處理,自定義類型處理器(TypeHandler)可以實(shí)現(xiàn)這一需求,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-10-10
  • 為什么阿里巴巴要求日期格式化時(shí)必須有使用y表示年

    為什么阿里巴巴要求日期格式化時(shí)必須有使用y表示年

    這篇文章主要介紹了為什么阿里巴巴要求日期格式化時(shí)必須有使用y表示年,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-08-08
  • springBoot2.X配置全局捕獲異常的操作

    springBoot2.X配置全局捕獲異常的操作

    這篇文章主要介紹了springBoot2.X配置全局捕獲異常的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • JVM參數(shù)-Xms和-Xmx的作用及說明

    JVM參數(shù)-Xms和-Xmx的作用及說明

    這篇文章主要介紹了JVM參數(shù)-Xms和-Xmx的作用及說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • IDEA快速生成實(shí)體類的示例教程

    IDEA快速生成實(shí)體類的示例教程

    這篇文章主要介紹了IDEA快速生成實(shí)體類的示例教程,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • Java函數(shù)式編程(六):Optional

    Java函數(shù)式編程(六):Optional

    這篇文章主要介紹了Java函數(shù)式編程(六):Optional,本文是系列文章的第6篇,其它文章請(qǐng)參閱本文底部的相關(guān)文章,需要的朋友可以參考下
    2014-09-09
  • SpringBoot2.3.0配置JPA的實(shí)現(xiàn)示例

    SpringBoot2.3.0配置JPA的實(shí)現(xiàn)示例

    這篇文章主要介紹了SpringBoot2.3.0配置JPA的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-08-08
  • Java 超詳細(xì)講解IO操作字節(jié)流與字符流

    Java 超詳細(xì)講解IO操作字節(jié)流與字符流

    本章具體介紹了字節(jié)流、字符流的基本使用方法,圖解穿插代碼實(shí)現(xiàn)。 JAVA從基礎(chǔ)開始講,后續(xù)會(huì)講到JAVA高級(jí),中間會(huì)穿插面試題和項(xiàng)目實(shí)戰(zhàn),希望能給大家?guī)韼椭?/div> 2022-03-03

最新評(píng)論