新聞中心
1.代理模式概述
成都創(chuàng)新互聯(lián)是一家集網(wǎng)站建設,圍場企業(yè)網(wǎng)站建設,圍場品牌網(wǎng)站建設,網(wǎng)站定制,圍場網(wǎng)站建設報價,網(wǎng)絡營銷,網(wǎng)絡優(yōu)化,圍場網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強企業(yè)競爭力??沙浞譂M足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時我們時刻保持專業(yè)、時尚、前沿,時刻以成就客戶成長自我,堅持不斷學習、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實用型網(wǎng)站。
為什么要有“代理”? 生活中就有很多例子,例如委托業(yè)務等等,代理就是被代理者沒有能力或者不愿意去完成某件事情,需要找個人代替自己去完成這件事,這才是“代理”存在的原因。例如,我現(xiàn)在需要出國,但是我不愿意自己去辦簽證、預定機票和酒店(覺得麻煩 ,那么就可以找旅行社去幫我辦,這時候旅行社就是代理,而我自己就是被代理了。
代理模式的定義:ProxyPattern(即:代理模式),23種常用的面向?qū)ο筌浖脑O計模式之一為其他對象提供一種代理以控制對這個對象的訪問。在某些情況下,一個對象不適合或者不能直接引用另一個對象,而代理對象可以在客戶端和目標對象之間起到中介的作用。
2.靜態(tài)代理
以下我們借用JinLian和XiMen的例子來介紹一下靜態(tài)代理
2.1不使用代理模式時
JinLian類的內(nèi)部有一個happy方法,此時如果我們的XiMen類如果想要使用這個方法,正常情況下直接創(chuàng)建一個JinLian類調(diào)用happy方法即可
public class JinLian{
public void happy() {
System.out.println("金蓮happy...");
}
}
public class XiMen {
public static void main(String[] args) {
JinLian jinLian =new JinLian();
jinLian.happy();
}
}
2.2使用靜態(tài)代理
如果我們使用靜態(tài)代理模式的話,XiMen類如果想要調(diào)用happy方法,那么它便不會直接去找JinLian類,而是通過一個代理對象WangPo間接找到JinLian類。這樣做的好處是,我們可以通過WangPo這個代理,在happy方法中添加一些想要的功能,例如:
public class WangPo {
JinLian jinLian ;
public WangPo(JinLian jinLian) {
this.jinLian = jinLian;
}
public void happy(){
openHouse();
jinLian.happy();
clear();
}
public void clear(){
System.out.println("打掃戰(zhàn)場...");
}
public void openHouse(){
System.out.println("以做衣服的名義把2人約到房間...");
}
}//WangPo
public class XiMen {
public static void main(String[] args) {
JinLian jinLian =new JinLian();
WangPo wangPo =new WangPo(jinLian);
wangPo.happy();
}
}//XiMen
public class JinLian {
public void happy() {
System.out.println("金蓮happy...");
}
}//JinLian
這樣,XiMen類想要調(diào)用happy方法,并且想要使用新增的功能時,只需要調(diào)用WangPo的happy方法即可。
2.3靜態(tài)代理改進
現(xiàn)在,雖然實現(xiàn)了代理模式,但是目前的代理對象和委托對象都已經(jīng)固定了,這樣做局限性很大,我們的XiMen類如果想要調(diào)用其它委托者的Happy方法,就只能找其它的代理對象,此時我們對代碼再次改進一下,新加一個FindHappy接口,委托者對象只需要實現(xiàn)FinHappy接口,便可以實現(xiàn)一個代理對象代理多個委托對象。如果,XiMen現(xiàn)在想要通過WangPo使用JinLian和YanPoXi的happy方法。我們可以做出如下改寫
public interface FindHappy {// 抽象角色
void happy();
}
public class JinLian implements FindHappy {//委托者1
@Override
public void happy() {
System.out.println("金蓮happy...");
}
}//JinLian
public class YanPoXi implements FindHappy {//委托者2
@Override
public void happy() {
System.out.println("閻婆惜happy...");
}
}//YanPoXi
public class WangPo implements FindHappy {//代理者
FindHappy findHappy;
public WangPo(FindHappy findHappy) {
this.findHappy = findHappy;
}
public void openHouse(){
System.out.println("以做衣服的名義把2人約到房間...");
}
@Override
public void happy() {
openHouse();
// 讓委托者happy()
findHappy.happy();
clear();
}
public void clear(){
System.out.println("打掃戰(zhàn)場...");
}
}
public class XiMen {
public static void main(String[] args) {
JinLian jinLian = new JinLian();
YanPoXi yanPoXi = new YanPoXi();
WangPo wangPo = new WangPo(yanPoXi);
WangPo wangPo1 =new WangPo(jinLian);
wangPo.happy();
wangPo1.happy();
}
}
如此一來我們便可以通過一個代理對象代理所有重寫了happy方法的對象
這便是靜態(tài)代理模式
3.動態(tài)代理
動態(tài)代理與靜態(tài)代理的區(qū)別是,動態(tài)代理的代理對象是在使用者調(diào)用委托者的方法時才創(chuàng)建的,動態(tài)代理它可以直接給某一個目標(被代理 對象)對象(實現(xiàn)了某個或者某些接口)生成一個代理對象,也就是當XiMen類想要使用happy方法時,代理對象才會生成,并且不需要代理類存在
3.1動態(tài)代理原理
動態(tài)代理與代理模式原理是一樣的,只是它沒有具體的代理類,直接通過反射生成了一個代理對象。簡而言之:動態(tài)代理就是直接通過反射生成一個代理對象,代理類是不需要存在的鄭州好的婦科醫(yī)院 http://www.zzkedayy.com/
3.2動態(tài)代理的獲取及使用
jdk提供一個Proxy類可以直接給實現(xiàn)接口類的對象直接生成代理對象
Proxy的API介紹
Proxy.newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)生成一個代理對象,這個方法的返回值類型是一個Object類型,但是在此方法的第三個參數(shù)的匿名內(nèi)部類中我們重寫了委托者的方法,而委托者又必定實現(xiàn)了FindHappy接口,因此此處我們可以使用向下強轉(zhuǎn),并用一個Findhappy接口的對象進行接收
參數(shù)1:ClassLoader loader 被代理對象的類加載器, 一般使用被代理對象的類加載器
參數(shù)2:Class[] interfaces 被代理對象的要實現(xiàn)的接口 一般使用的被代理對象實現(xiàn)的接口
參數(shù)3:InvocationHandler h (接口)(使用匿名內(nèi)部類直接實現(xiàn))執(zhí)行處理類返回值: 代理對象
InvocationHandler中的invoke(Object proxy, Method method, Object[] args)方法:調(diào)用代理類的任何方法,此方法都會執(zhí)行,因此當一個委托對象中有多個方法需要代理時,需要在此匿名內(nèi)部類中使用反射獲取方法名,判斷是哪個方法在請求代理,一一實現(xiàn)各個方法需要增添的功能
3.1:代理對象(慎用)參數(shù)
3.2:當前執(zhí)行的方法參數(shù)
3.3:當前執(zhí)行的方法運行時傳遞過來的參數(shù)返回值:當前方法執(zhí)行的返回值
public class Demo1 {
public static void main(String[] args) {
// 動態(tài)產(chǎn)生一個JinLian類的代理對象 類似生成王婆代理對象
FindHappy proxy = (FindHappy) Proxy.newProxyInstance(JinLian.class.getClassLoader(), JinLian.class.getInterfaces(), new InvocationHandler() {
public void openHouse(){
System.out.println("以做衣服的名義把2人約到房間...");
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 但凡是代理對象調(diào)用方法,這個invoke方法就會執(zhí)行
// 參數(shù)1proxy:代理對象(慎用),因為代理對象調(diào)用方法就會執(zhí)行invoke,很容易出現(xiàn)遞歸,出現(xiàn)棧內(nèi)存溢出
// 參數(shù)2 method:當前代理對象調(diào)用的方法
// 參數(shù)3:當前代理對象調(diào)用的方法,運行時傳遞過來的參數(shù)
// 返回值:當前方法執(zhí)行的返回值
if(method.getName()=="happy"){
openHouse();
method.invoke(JinLian.class.newInstance());// 類似 jinLian.happy()
clear();
return null;
}
}
public void clear(){
System.out.println("打掃戰(zhàn)場...");
}
});
// 使用代理對象調(diào)用方法
proxy.happy();
}
網(wǎng)頁標題:動態(tài)代理的理解
URL鏈接:http://fisionsoft.com.cn/article/jsosoc.html