在代理模式(Proxy Pattern)中,一個(gè)類(lèi)代表另一個(gè)類(lèi)的功能。這種類(lèi)型的設(shè)計(jì)模式屬于結(jié)構(gòu)型模式。
在代理模式中,我們創(chuàng)建具有現(xiàn)有對(duì)象的對(duì)象,以便向外界提供功能接口。
介紹
意圖:為其他對(duì)象提供一種代理以控制對(duì)這個(gè)對(duì)象的訪(fǎng)問(wèn)。
主要解決:在直接訪(fǎng)問(wèn)對(duì)象時(shí)帶來(lái)的問(wèn)題,比如說(shuō):要訪(fǎng)問(wèn)的對(duì)象在遠(yuǎn)程的機(jī)器上。在面向?qū)ο笙到y(tǒng)中,有些對(duì)象由于某些原因(比如對(duì)象創(chuàng)建開(kāi)銷(xiāo)很大,或者某些操作需要安全控制,或者需要進(jìn)程外的訪(fǎng)問(wèn)),直接訪(fǎng)問(wèn)會(huì)給使用者或者系統(tǒng)結(jié)構(gòu)帶來(lái)很多麻煩,我們可以在訪(fǎng)問(wèn)此對(duì)象時(shí)加上一個(gè)對(duì)此對(duì)象的訪(fǎng)問(wèn)層。
何時(shí)使用:想在訪(fǎng)問(wèn)一個(gè)類(lèi)時(shí)做一些控制。
如何解決:增加中間層。
關(guān)鍵代碼:實(shí)現(xiàn)與被代理類(lèi)組合。
應(yīng)用實(shí)例: 1、Windows 里面的快捷方式。 2、豬八戒去找高翠蘭結(jié)果是孫悟空變的,可以這樣理解:把高翠蘭的外貌抽象出來(lái),高翠蘭本人和孫悟空都實(shí)現(xiàn)了這個(gè)接口,豬八戒訪(fǎng)問(wèn)高翠蘭的時(shí)候看不出來(lái)這個(gè)是孫悟空,所以說(shuō)孫悟空是高翠蘭代理類(lèi)。 3、買(mǎi)火車(chē)票不一定在火車(chē)站買(mǎi),也可以去代售點(diǎn)。 4、一張支票或銀行存單是賬戶(hù)中資金的代理。支票在市場(chǎng)交易中用來(lái)代替現(xiàn)金,并提供對(duì)簽發(fā)人賬號(hào)上資金的控制。 5、spring aop。
優(yōu)點(diǎn): 1、職責(zé)清晰。 2、高擴(kuò)展性。 3、智能化。
缺點(diǎn): 1、由于在客戶(hù)端和真實(shí)主題之間增加了代理對(duì)象,因此有些類(lèi)型的代理模式可能會(huì)造成請(qǐng)求的處理速度變慢。 2、實(shí)現(xiàn)代理模式需要額外的工作,有些代理模式的實(shí)現(xiàn)非常復(fù)雜。
使用場(chǎng)景:按職責(zé)來(lái)劃分,通常有以下使用場(chǎng)景: 1、遠(yuǎn)程代理。 2、虛擬代理。 3、Copy-on-Write 代理。 4、保護(hù)(Protect or Access)代理。 5、Cache代理。 6、防火墻(Firewall)代理。 7、同步化(Synchronization)代理。 8、智能引用(Smart Reference)代理。
注意事項(xiàng): 1、和適配器模式的區(qū)別:適配器模式主要改變所考慮對(duì)象的接口,而代理模式不能改變所代理類(lèi)的接口。 2、和裝飾器模式的區(qū)別:裝飾器模式為了增強(qiáng)功能,而代理模式是為了加以控制。
實(shí)現(xiàn)
我們將創(chuàng)建一個(gè) Image 接口和實(shí)現(xiàn)了 Image 接口的實(shí)體類(lèi)。ProxyImage 是一個(gè)代理類(lèi),減少 RealImage 對(duì)象加載的內(nèi)存占用。
ProxyPatternDemo,我們的演示類(lèi)使用 ProxyImage 來(lái)獲取要加載的 Image 對(duì)象,并按照需求進(jìn)行顯示。
步驟 1
創(chuàng)建一個(gè)接口。
Image.java
public interface Image {
void display();
}
步驟 2
創(chuàng)建實(shí)現(xiàn)接口的實(shí)體類(lèi)。
RealImage.java
public class RealImage implements Image {
private String fileName;
public RealImage(String fileName){
this.fileName = fileName;
loadFromDisk(fileName);
}
@Override
public void display() {
System.out.println("Displaying " + fileName);
}
private void loadFromDisk(String fileName){
System.out.println("Loading " + fileName);
}
}
ProxyImage.java
public class ProxyImage implements Image{
private RealImage realImage;
private String fileName;
public ProxyImage(String fileName){
this.fileName = fileName;
}
@Override
public void display() {
if(realImage == null){
realImage = new RealImage(fileName);
}
realImage.display();
}
}
步驟 3
當(dāng)被請(qǐng)求時(shí),使用 ProxyImage 來(lái)獲取 RealImage 類(lèi)的對(duì)象。
ProxyPatternDemo.java
public class ProxyPatternDemo {
public static void main(String[] args) {
Image image = new ProxyImage("test_10mb.jpg");
//圖像將從磁盤(pán)加載
image.display();
System.out.println("");
//圖像將無(wú)法從磁盤(pán)加載
image.display();
}
}
步驟 4
驗(yàn)證輸出。
Loading test_10mb.jpg
Displaying test_10mb.jpg
Displaying test_10mb.jpg