国产av日韩一区二区三区精品,成人性爱视频在线观看,国产,欧美,日韩,一区,www.成色av久久成人,2222eeee成人天堂

首頁(yè) Java Java入門(mén) java的三種代理模式分別是什麼

java的三種代理模式分別是什麼

Jan 30, 2021 am 09:44 AM
java 代理模式

java的三種代理模式分別是什麼

首先來(lái)簡(jiǎn)單說(shuō)明一下什麼是代理模式。

代理(Proxy)是一種設(shè)計(jì)模式,提供了對(duì)目標(biāo)物件另外的存取方式;即透過(guò)代理物件存取目標(biāo)物件.這樣做的好處是:可以在目標(biāo)物件實(shí)現(xiàn)的基礎(chǔ)上,增強(qiáng)額外的功能操作,即擴(kuò)展目標(biāo)對(duì)象的功能.
這裡使用到編程中的一個(gè)思想:不要隨意去修改別人已經(jīng)寫(xiě)好的代碼或者方法,如果需改修改,可以通過(guò)代理的方式來(lái)擴(kuò)展該方法

舉個(gè)例子來(lái)說(shuō)明代理的作用:假設(shè)我們想邀請(qǐng)一位明星,那麼並不是直接連接明星,而是聯(lián)繫明星的經(jīng)紀(jì)人,來(lái)達(dá)到同樣的目的.明星就是一個(gè)目標(biāo)對(duì)象,他只要負(fù)責(zé)活動(dòng)中的節(jié)目,而其他瑣碎的事情就交給他的代理人(經(jīng)紀(jì)人)來(lái)解決.這就是代理思想在現(xiàn)實(shí)中的一個(gè)例子

用圖表示如下:

java的三種代理模式分別是什麼

代理模式的關(guān)鍵點(diǎn)是:代理物件與目標(biāo)物件.代理物件是對(duì)目標(biāo)物件的擴(kuò)充,並會(huì)呼叫目標(biāo)物件

1.1.靜態(tài)代理

靜態(tài)代理程式在使用時(shí),需要定義介面或父類(lèi)別,被代理物件與代理物件一起實(shí)作相同的介面或是繼承相同父類(lèi)別.

下面舉個(gè)案例來(lái)解釋:
模擬儲(chǔ)存動(dòng)作,定義一個(gè)儲(chǔ)存動(dòng)作的介面:IUserDao.java,然後目標(biāo)物件實(shí)作這個(gè)介面的方法UserDao.java,此時(shí)如果使用靜態(tài)代理程式方式,就需要在代理物件(UserDaoProxy.java)中也實(shí)作IUserDao介面.呼叫的時(shí)候透過(guò)呼叫代理物件的方法來(lái)呼叫目標(biāo)物件.
要注意的是,代理物件與目標(biāo)物件要實(shí)現(xiàn)相同的介面,然後透過(guò)呼叫相同的方法來(lái)呼叫目標(biāo)物件的方法

程式碼範(fàn)例:
介面:IUserDao.java

/**
 * 接口
 */public interface IUserDao {    void save();
}

目標(biāo)物件:UserDao.java

/**
 * 接口實(shí)現(xiàn)
 * 目標(biāo)對(duì)象
 */public class UserDao implements IUserDao {    public void save() {
        System.out.println("----已經(jīng)保存數(shù)據(jù)!----");
    }
}

代理物件:UserDaoProxy.java

/**
 * 代理對(duì)象,靜態(tài)代理
 */public class UserDaoProxy implements IUserDao{    //接收保存目標(biāo)對(duì)象
    private IUserDao target;    public UserDaoProxy(IUserDao target){        this.target=target;
    }    public void save() {
        System.out.println("開(kāi)始事務(wù)...");
        target.save();//執(zhí)行目標(biāo)對(duì)象的方法
        System.out.println("提交事務(wù)...");
    }
}

(學(xué)習(xí)影片分享:java影片教學(xué)

測(cè)試類(lèi)別:App.java

/**
 * 測(cè)試類(lèi)
 */public class App {    public static void main(String[] args) {        //目標(biāo)對(duì)象
        UserDao target = new UserDao();        //代理對(duì)象,把目標(biāo)對(duì)象傳給代理對(duì)象,建立代理關(guān)系
        UserDaoProxy proxy = new UserDaoProxy(target);

        proxy.save();//執(zhí)行的是代理的方法
    }
}

靜態(tài)代理程式總結(jié):
1.可以做到在不修改目標(biāo)物件的功能前提下,對(duì)目標(biāo)功能擴(kuò)展.
2.缺點(diǎn):

因?yàn)榇砦锛枰c目標(biāo)物件實(shí)現(xiàn)一樣的介面,所以會(huì)有很多代理類(lèi)別,類(lèi)別太多.同時(shí),一旦介面增加方法,目標(biāo)物件與代理物件都要維護(hù).

如何解決靜態(tài)代理程式中的缺點(diǎn)呢?答案是可以使用動(dòng)態(tài)代理方式

#1.2.動(dòng)態(tài)代理

動(dòng)態(tài)代理有以下特點(diǎn):
1.代理對(duì)象,不需要實(shí)現(xiàn)接口
2.代理對(duì)象的生成,是利用JDK的API,動(dòng)態(tài)的在內(nèi)存中構(gòu)建代理對(duì)象(需要我們指定創(chuàng)建代理對(duì)象/目標(biāo)物件實(shí)現(xiàn)的介面的類(lèi)型)
3.動(dòng)態(tài)代理也叫做:JDK代理,介面代理

JDK中產(chǎn)生代理物件的API
代理類(lèi)別所在套件:java.lang.reflect .Proxy
JDK實(shí)作代理只需要使用newProxyInstance方法,但是該方法需要接收三個(gè)參數(shù),完整的寫(xiě)法是:

static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )

注意該方法是在Proxy類(lèi)別中是靜態(tài)方法,且接收的三個(gè)參數(shù)依序?yàn)?

ClassLoader loader,:指定目前目標(biāo)物件使用類(lèi)別載入器,取得載入器的方法是固定的

Class[] interfaces,:目標(biāo)物件實(shí)現(xiàn)的介面的類(lèi)型,使用泛型方式確認(rèn)類(lèi)型

InvocationHandler h:事件處理,執(zhí)行目標(biāo)物件的方法時(shí),會(huì)觸發(fā)事件處理器的方法,會(huì)把目前執(zhí)行目標(biāo)物件的方法作為參數(shù)傳入

程式碼範(fàn)例:
介面類(lèi)別IUserDao.java以及介面實(shí)作類(lèi)別,目標(biāo)物件UserDao是一樣的,沒(méi)有做修改.在這個(gè)基礎(chǔ)上,增加一個(gè)代理工廠(chǎng)類(lèi)別(ProxyFactory.java) ,將代理類(lèi)別寫(xiě)在這個(gè)地方,然後在測(cè)試類(lèi)別(需要使用到代理的程式碼)中先建立目標(biāo)物件和代理物件的聯(lián)繫,然後代用代理物件的中同名方法

代理工廠(chǎng)類(lèi)別: ProxyFactory.java

/**
 * 創(chuàng)建動(dòng)態(tài)代理對(duì)象
 * 動(dòng)態(tài)代理不需要實(shí)現(xiàn)接口,但是需要指定接口類(lèi)型
 */public class ProxyFactory{    //維護(hù)一個(gè)目標(biāo)對(duì)象
    private Object target;    public ProxyFactory(Object target){        this.target=target;
    }   //給目標(biāo)對(duì)象生成代理對(duì)象
    public Object getProxyInstance(){        return Proxy.newProxyInstance(
                target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),                new InvocationHandler() {                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        System.out.println("開(kāi)始事務(wù)2");                        //執(zhí)行目標(biāo)對(duì)象方法
                        Object returnValue = method.invoke(target, args);
                        System.out.println("提交事務(wù)2");                        return returnValue;
                    }
                }
        );
    }

}

測(cè)試類(lèi)別:App.java

/**
 * 測(cè)試類(lèi)
 */public class App {    public static void main(String[] args) {        // 目標(biāo)對(duì)象
        IUserDao target = new UserDao();        // 【原始的類(lèi)型 class cn.itcast.b_dynamic.UserDao】
        System.out.println(target.getClass());        // 給目標(biāo)對(duì)象,創(chuàng)建代理對(duì)象
        IUserDao proxy = (IUserDao) new ProxyFactory(target).getProxyInstance();        // class $Proxy0   內(nèi)存中動(dòng)態(tài)生成的代理對(duì)象
        System.out.println(proxy.getClass());        // 執(zhí)行方法   【代理對(duì)象】
        proxy.save();
    }
}

總結(jié):
代理物件不需要實(shí)作介面,但是目標(biāo)物件一定要實(shí)作介面,否則不能用動(dòng)態(tài)代理

1.3.Cglib代理

上面的靜態(tài)代理和動(dòng)態(tài)代理模式都是要求目標(biāo)對(duì)像是實(shí)現(xiàn)一個(gè)接口的目標(biāo)對(duì)象,但是有時(shí)候目標(biāo)對(duì)像只是一個(gè)單獨(dú)的對(duì)象,並沒(méi)有實(shí)現(xiàn)任何的介面,這個(gè)時(shí)候就可以使用以目標(biāo)物件子類(lèi)別的方式類(lèi)別實(shí)作代理,這種方法就叫做:Cglib代理

Cglib代理,也叫作子類(lèi)代理,它是在內(nèi)存中構(gòu)建一個(gè)子類(lèi)對(duì)象從而實(shí)現(xiàn)對(duì)目標(biāo)對(duì)象功能的擴(kuò)展.

JDK的動(dòng)態(tài)代理有一個(gè)限制,就是使用動(dòng)態(tài)代理的對(duì)象必須實(shí)現(xiàn)一個(gè)或多個(gè)接口,如果想代理沒(méi)有實(shí)現(xiàn)接口的類(lèi),就可以使用Cglib實(shí)現(xiàn).Cglib是一個(gè)強(qiáng)大的高性能的代碼生成包,它可以在運(yùn)行期擴(kuò)展java類(lèi)與實(shí)現(xiàn)java接口.它廣泛的被許多AOP的框架使用,例如Spring AOP和synaop,為他們提供方法的interception(攔截)Cglib包的底層是通過(guò)使用一個(gè)小而塊的字節(jié)碼處理框架ASM來(lái)轉(zhuǎn)換字節(jié)碼并生成新的類(lèi).不鼓勵(lì)直接使用ASM,因?yàn)樗竽惚仨殞?duì)JVM內(nèi)部結(jié)構(gòu)包括class文件的格式和指令集都很熟悉.

Cglib子類(lèi)代理實(shí)現(xiàn)方法:
1.需要引入cglib的jar文件,但是Spring的核心包中已經(jīng)包括了Cglib功能,所以直接引入pring-core-3.2.5.jar即可.
2.引入功能包后,就可以在內(nèi)存中動(dòng)態(tài)構(gòu)建子類(lèi)
3.代理的類(lèi)不能為final,否則報(bào)錯(cuò)
4.目標(biāo)對(duì)象的方法如果為final/static,那么就不會(huì)被攔截,即不會(huì)執(zhí)行目標(biāo)對(duì)象額外的業(yè)務(wù)方法.

代碼示例:
目標(biāo)對(duì)象類(lèi):UserDao.java

/**
 * 目標(biāo)對(duì)象,沒(méi)有實(shí)現(xiàn)任何接口
 */public class UserDao {    public void save() {
        System.out.println("----已經(jīng)保存數(shù)據(jù)!----");
    }
}

Cglib代理工廠(chǎng):ProxyFactory.java

/**
 * Cglib子類(lèi)代理工廠(chǎng)
 * 對(duì)UserDao在內(nèi)存中動(dòng)態(tài)構(gòu)建一個(gè)子類(lèi)對(duì)象
 */public class ProxyFactory implements MethodInterceptor{    //維護(hù)目標(biāo)對(duì)象
    private Object target;    public ProxyFactory(Object target) {        this.target = target;
    }    //給目標(biāo)對(duì)象創(chuàng)建一個(gè)代理對(duì)象
    public Object getProxyInstance(){        //1.工具類(lèi)
        Enhancer en = new Enhancer();        //2.設(shè)置父類(lèi)
        en.setSuperclass(target.getClass());        //3.設(shè)置回調(diào)函數(shù)
        en.setCallback(this);        //4.創(chuàng)建子類(lèi)(代理對(duì)象)
        return en.create();

    }    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("開(kāi)始事務(wù)...");        //執(zhí)行目標(biāo)對(duì)象的方法
        Object returnValue = method.invoke(target, args);

        System.out.println("提交事務(wù)...");        return returnValue;
    }
}

測(cè)試類(lèi):

/**
 * 測(cè)試類(lèi)
 */public class App {    @Test
    public void test(){        //目標(biāo)對(duì)象
        UserDao target = new UserDao();        //代理對(duì)象
        UserDao proxy = (UserDao)new ProxyFactory(target).getProxyInstance();        //執(zhí)行代理對(duì)象的方法
        proxy.save();
    }
}

在Spring的AOP編程中:
如果加入容器的目標(biāo)對(duì)象有實(shí)現(xiàn)接口,用JDK代理
如果目標(biāo)對(duì)象沒(méi)有實(shí)現(xiàn)接口,用Cglib代理

相關(guān)推薦:java入門(mén)教程

以上是java的三種代理模式分別是什麼的詳細(xì)內(nèi)容。更多資訊請(qǐng)關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

本網(wǎng)站聲明
本文內(nèi)容由網(wǎng)友自願(yuàn)投稿,版權(quán)歸原作者所有。本站不承擔(dān)相應(yīng)的法律責(zé)任。如發(fā)現(xiàn)涉嫌抄襲或侵權(quán)的內(nèi)容,請(qǐng)聯(lián)絡(luò)admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費(fèi)脫衣圖片

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅(qū)動(dòng)的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線(xiàn)上人工智慧工具。

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費(fèi)的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費(fèi)的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強(qiáng)大的PHP整合開(kāi)發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺(jué)化網(wǎng)頁(yè)開(kāi)發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級(jí)程式碼編輯軟體(SublimeText3)

熱門(mén)話(huà)題

如何在Java的地圖上迭代? 如何在Java的地圖上迭代? Jul 13, 2025 am 02:54 AM

遍歷Java中的Map有三種常用方法:1.使用entrySet同時(shí)獲取鍵和值,適用於大多數(shù)場(chǎng)景;2.使用keySet或values分別遍歷鍵或值;3.使用Java8的forEach簡(jiǎn)化代碼結(jié)構(gòu)。 entrySet返回包含所有鍵值對(duì)的Set集合,每次循環(huán)獲取Map.Entry對(duì)象,適合頻繁訪(fǎng)問(wèn)鍵和值的情況;若只需鍵或值,可分別調(diào)用keySet()或values(),也可在遍歷鍵時(shí)通過(guò)map.get(key)獲取值;Java8中可通過(guò)Lambda表達(dá)式使用forEach((key,value)-&gt

Java可選示例 Java可選示例 Jul 12, 2025 am 02:55 AM

Optional能清晰表達(dá)意圖並減少null判斷的代碼噪音。 1.Optional.ofNullable是處理可能為null對(duì)象的常用方式,如從map中取值時(shí)可結(jié)合orElse提供默認(rèn)值,邏輯更清晰簡(jiǎn)潔;2.通過(guò)鍊式調(diào)用map實(shí)現(xiàn)嵌套取值,安全地避免NPE,任一環(huán)節(jié)為null則自動(dòng)終止並返回默認(rèn)值;3.filter可用於條件篩選,滿(mǎn)足條件才繼續(xù)執(zhí)行後續(xù)操作,否則直接跳到o??rElse,適合輕量級(jí)業(yè)務(wù)判斷;4.不建議過(guò)度使用Optional,如基本類(lèi)型或簡(jiǎn)單邏輯中其反而增加複雜度,部分場(chǎng)景直接返回nu

如何修復(fù)java.io.notserializable Exception? 如何修復(fù)java.io.notserializable Exception? Jul 12, 2025 am 03:07 AM

遇到j(luò)ava.io.NotSerializableException的核心解決方法是確保所有需序列化的類(lèi)實(shí)現(xiàn)Serializable接口,並檢查嵌套對(duì)象的序列化支持。 1.給主類(lèi)添加implementsSerializable;2.確保類(lèi)中自定義字段對(duì)應(yīng)的類(lèi)也實(shí)現(xiàn)Serializable;3.用transient標(biāo)記不需要序列化的字段;4.檢查集合或嵌套對(duì)像中的非序列化類(lèi)型;5.查看異常信息定位具體哪個(gè)類(lèi)未實(shí)現(xiàn)接口;6.對(duì)無(wú)法修改的類(lèi)考慮替換設(shè)計(jì),如保存關(guān)鍵數(shù)據(jù)或使用可序列化的中間結(jié)構(gòu);7.考慮改

Java中的可比較與比較器 Java中的可比較與比較器 Jul 13, 2025 am 02:31 AM

在Java中,Comparable用於類(lèi)內(nèi)部定義默認(rèn)排序規(guī)則,Comparator用於外部靈活定義多種排序邏輯。 1.Comparable是類(lèi)自身實(shí)現(xiàn)的接口,通過(guò)重寫(xiě)compareTo()方法定義自然順序,適用於類(lèi)有固定、最常用的排序方式,如String或Integer。 2.Comparator是外部定義的函數(shù)式接口,通過(guò)compare()方法實(shí)現(xiàn),適合同一類(lèi)需要多種排序方式、無(wú)法修改類(lèi)源碼或排序邏輯經(jīng)常變化的情況。兩者區(qū)別在於Comparable只能定義一種排序邏輯且需修改類(lèi)本身,而Compar

如何處理Java中的字符編碼問(wèn)題? 如何處理Java中的字符編碼問(wèn)題? Jul 13, 2025 am 02:46 AM

處理Java中的字符編碼問(wèn)題,關(guān)鍵是在每一步都明確指定使用的編碼。 1.讀寫(xiě)文本時(shí)始終指定編碼,使用InputStreamReader和OutputStreamWriter並傳入明確的字符集,避免依賴(lài)系統(tǒng)默認(rèn)編碼。 2.在網(wǎng)絡(luò)邊界處理字符串時(shí)確保兩端一致,設(shè)置正確的Content-Type頭並用庫(kù)顯式指定編碼。 3.謹(jǐn)慎使用String.getBytes()和newString(byte[]),應(yīng)始終手動(dòng)指定StandardCharsets.UTF_8以避免平臺(tái)差異導(dǎo)致的數(shù)據(jù)損壞??傊ㄟ^(guò)在每個(gè)階段

Java方法參考解釋了 Java方法參考解釋了 Jul 12, 2025 am 02:59 AM

方法引用是Java中一種簡(jiǎn)化Lambda表達(dá)式的寫(xiě)法,使代碼更簡(jiǎn)潔。它不是新語(yǔ)法,而是Java8引入的Lambda表達(dá)式的一種快捷方式,適用於函數(shù)式接口的上下文。其核心在於將已有方法直接作為函數(shù)式接口的實(shí)現(xiàn)來(lái)使用。例如System.out::println等價(jià)於s->System.out.println(s)。方法引用主要有四種形式:1.靜態(tài)方法引用(ClassName::staticMethodName);2.實(shí)例方法引用(綁定到特定對(duì)象,instance::methodName);3.

如何在Java解析JSON? 如何在Java解析JSON? Jul 11, 2025 am 02:18 AM

解析JSON在Java中的常見(jiàn)方式有三種:使用Jackson、Gson或org.json。 1.Jackson適合大多數(shù)項(xiàng)目,性能好且功能全面,支持對(duì)象與JSON字符串之間的轉(zhuǎn)換及註解映射;2.Gson更適合Android項(xiàng)目或輕量級(jí)需求,使用簡(jiǎn)單但處理複雜結(jié)構(gòu)和高性能場(chǎng)景略遜;3.org.json適用於簡(jiǎn)單任務(wù)或小腳本,不推薦用於大型項(xiàng)目,因其靈活性和類(lèi)型安全不足。選擇應(yīng)根據(jù)實(shí)際需求決定。

新電子郵件的Outlook快捷方式 新電子郵件的Outlook快捷方式 Jul 11, 2025 am 03:25 AM

在Outlook中快速新建郵件的方法如下:1.桌面版使用快捷鍵Ctrl Shift M,可直接彈出新郵件窗口;2.網(wǎng)頁(yè)版可通過(guò)創(chuàng)建包含JavaScript的書(shū)籤(如javascript:document.querySelector("divrole='button'").click())實(shí)現(xiàn)一鍵新建郵件;3.使用瀏覽器插件(如Vimium、CrxMouseGestures)自定義快捷鍵觸發(fā)“新建郵件”按鈕;4.Windows用戶(hù)還可通過(guò)右鍵任務(wù)欄Outlook圖標(biāo)選擇“新建電

See all articles