代理(Proxy)模式:
我們一直強(qiáng)調(diào)成都做網(wǎng)站、網(wǎng)站制作、成都外貿(mào)網(wǎng)站建設(shè)對于企業(yè)的重要性,如果您也覺得重要,那么就需要我們慎重對待,選擇一個安全靠譜的網(wǎng)站建設(shè)公司,企業(yè)網(wǎng)站我們建議是要么不做,要么就做好,讓網(wǎng)站能真正成為企業(yè)發(fā)展過程中的有力推手。專業(yè)網(wǎng)絡(luò)公司不一定是大公司,成都創(chuàng)新互聯(lián)作為專業(yè)的網(wǎng)絡(luò)公司選擇我們就是放心。
代理模式是對象的結(jié)構(gòu)模式。代理模式給某一個對象提供一個代理對象,并由代理對象控制對原對象的引用。
核心作用:
通過代理,控制對對象的訪問。
可以詳細(xì)控制訪問某個(某類)對象的方法,在調(diào)用這個方法之前做前置處理,在調(diào)用這個方法之后做后置處理。即AOP的微觀實現(xiàn)。
AOP(Aspect Oriented Programming面向切面編程)的核心實現(xiàn)機(jī)制。
生活中的場景:
所謂代理,就是一個人或者機(jī)構(gòu)代表另一個人或者機(jī)構(gòu)采取行動。在一些情況下,一個客戶不想或者不能夠直接引用一個對象,而代理對象可以在客戶端和目標(biāo)對象之間起到中介的作用。
例如,客戶要找某個明星唱歌,首先需要找到他的經(jīng)紀(jì)人,然后由他的經(jīng)紀(jì)人來安排某明星唱歌。
經(jīng)紀(jì)人在演唱會舉辦前需要處理一些前置處理(面談,合同起草,簽字,收預(yù)付款,安排機(jī)票和車輛等)在演唱會舉辦后需要處理一些后置處理(收尾款等)。此時某明星(真實角色)只需要關(guān)心如何唱歌就可以了,其他事情全部交給經(jīng)紀(jì)人(代理)來辦。
核心角色:
這里寫圖片描述
抽象對象角色:聲明了代理對象和真實對象的共同接口,定義代理對象和真實對象的公共對外方法。這樣一來在任何可以使用真實對象的地方都可以使用代理對象。
真實對象角色:定義了代理對象所代表的真實對象。實現(xiàn)抽象對象,定義真實對象所需要實現(xiàn)的業(yè)務(wù)邏輯,供代理對象調(diào)用。關(guān)注真正的業(yè)務(wù)邏輯。
代理對象角色:實現(xiàn)抽象對象,是真實對象的代理,通過真實對象的業(yè)務(wù)邏輯方法來實現(xiàn)抽象方法,并附加自己的操作。將統(tǒng)一的流程控制放到代理對象中處理。
代理對象內(nèi)部含有真實對象的引用,從而可以在任何時候操作真實對象;代理對象提供一個與真實對象相同的接口,以便可以在任何時候替代真實對象。代理對象通常在客戶端調(diào)用傳遞給真實對象之前或之后,執(zhí)行某個操作,而不是單純地將調(diào)用傳遞給真實對象。
應(yīng)用場景:
安全代理:屏蔽對真實角色的直接訪問。
遠(yuǎn)程代理:通過代理類處理遠(yuǎn)程方法調(diào)用。
延遲加載:先加載輕量級的代理對象,真正需要再去加載真實對象。(圖片的延遲加載)
分類:
靜態(tài)代理:(靜態(tài)定義代理類)
上面例子的代碼:
1、聲明了代理對象和真實對象的共同接口,定義代理對象和真實對象的公共對外方法。
public interface Star { /** * 面談 */ void confer(); /** * 簽合同 */ void signContract(); /** * 訂票 */ void bookTicket(); /** * 唱歌 */ void sing(); /** * 收錢 */ void collectMoney(); }
2、定義一個真實對象的類,實現(xiàn)抽象接口所提供的方法。
public class RealStar implements Star { @Override public void bookTicket() { System.out.println("RealStar.bookTicket()"); } @Override public void collectMoney() { System.out.println("RealStar.collectMoney()"); } @Override public void confer() { System.out.println("RealStar.confer()"); } @Override public void signContract() { System.out.println("RealStar.signContract()"); } @Override public void sing() { System.out.println("RealStar.sing()"); } }
3、定義一個代理對象的類,實現(xiàn)抽象接口所提供的方法,并持有真實對象的引用。
public class ProxyStar implements Star{ private Star star; public ProxyStar(Star star) { super(); this.star = star; } @Override public void bookTicket() { System.out.println("ProxyStar.bookTicket()"); } @Override public void collectMoney() { System.out.println("ProxyStar.collectMoney()"); } @Override public void confer() { System.out.println("ProxyStar.confer()"); } @Override public void signContract() { System.out.println("ProxyStar.signContract()"); } @Override public void sing() { star.sing(); } }
4.測試類
public class Client { public static void main(String[] args) { //定義真實對象角色 Star realStar = new RealStar(); //定義代理對象角色,內(nèi)部含有真實對象的引用 Star proxyStar = new ProxyStar(realStar); proxyStar.confer(); proxyStar.signContract(); proxyStar.bookTicket(); proxyStar.sing(); proxyStar.collectMoney(); } }
運行結(jié)果如下:
ProxyStar.confer() ProxyStar.signContract() ProxyStar.bookTicket() RealStar.sing() ProxyStar.collectMoney()
從上面的例子可以看出代理對象將客戶端的調(diào)用委派給真實對象,在調(diào)用目標(biāo)對象的方法之前跟之后都可以執(zhí)行特定的操作。
動態(tài)代理:(動態(tài)生成代理類):
動態(tài)代理相比于靜態(tài)代理的優(yōu)點:
抽象角色中(接口)聲明的所有方法都被轉(zhuǎn)移到調(diào)用服務(wù)器的一個集中的方法中處理,這樣我們可以更加靈活和統(tǒng)一的處理眾多的方法。
JDK自帶的動態(tài)代理
java.lang.reflect.Proxy
動態(tài)生成代理類和對象
java.lang.reflect.InvocationHandler(處理器接口)
可以通過invoke方法實現(xiàn)對真實角色的代理訪問
每次通過Proxy生成代理類對象時都要指定對應(yīng)的處理器對象
測試代碼如下:
1、聲明了代理對象和真實對象的共同接口,定義代理對象和真實對象的公共對外方法。
public interface Star { /** * 面談 */ void confer(); /** * 簽合同 */ void signContract(); /** * 訂票 */ void bookTicket(); /** * 唱歌 */ void sing(); /** * 收錢 */ void collectMoney(); }
2、定義一個真實對象的類,實現(xiàn)抽象接口所提供的方法。
public class RealStar implements Star { @Override public void bookTicket() { System.out.println("RealStar.bookTicket()"); } @Override public void collectMoney() { System.out.println("RealStar.collectMoney()"); } @Override public void confer() { System.out.println("RealStar.confer()"); } @Override public void signContract() { System.out.println("RealStar.signContract()"); } @Override public void sing() { System.out.println("RealStar.sing()"); } }
3.定義一個StarHandler類實現(xiàn)InvocationHandler處理器接口,可以通過invoke方法實現(xiàn)對真實角色的代理訪問,也可在invoke方法中對眾多操作做統(tǒng)一的處理。
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; public class StarHandler implements InvocationHandler{ private Star realStar; public StarHandler(Star realStar) { super(); this.realStar = realStar; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //返回值 Object object = null; System.out.println("真正的方法執(zhí)行前!"); System.out.println("面談,簽合同,預(yù)付款,訂機(jī)票"); if(method.getName().equals("sing")){ object = method.invoke(realStar, args); } System.out.println("真正的方法執(zhí)行后!"); System.out.println("收尾款"); return object; } }
4.客戶端測試類
import java.lang.reflect.Proxy; public class Client { public static void main(String[] args) { Star realStar = new RealStar(); StarHandler handler = new StarHandler(realStar); //通過Proxy生成代理類對象并指定對應(yīng)的處理器對象 Star proxyStar = (Star)Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), new Class[]{Star.class}, handler); proxyStar.sing(); }
運行結(jié)果如下:
真正的方法執(zhí)行前!
面談,簽合同,預(yù)付款,訂機(jī)票
RealStar.sing()
真正的方法執(zhí)行后!
收尾款
開發(fā)框架中的應(yīng)用場景
代理模式在開發(fā)框架中的應(yīng)用場景是非常多的,實際上隨便選擇一個開發(fā)框架都有用到代理模式。例如:
mybatis中實現(xiàn)攔截器插件
AspectJ的實現(xiàn)
spring中AOP的實現(xiàn)
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。
本文題目:java設(shè)計模式筆記之代理模式
分享鏈接:http://jinyejixie.com/article6/igoeig.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供外貿(mào)網(wǎng)站建設(shè)、網(wǎng)站排名、網(wǎng)站策劃、網(wǎng)站改版、面包屑導(dǎo)航、網(wǎng)頁設(shè)計公司
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)