Java設計模式之代理模式(Proxy)


參考http://blog.csdn.net/jianghuxiaoxiami/article/details/3403924



1.代理模式

 

代理模式的作用是:爲其他對象提供一種代理以控制對這個對象的訪問。在某些情況下,一個客戶不想或者不能直接引用另一個對象,而代理對象可以在客戶端和目標對象之間起到中介的作用。

 

代理模式一般涉及到的角色有:

 

抽象角色:聲明真實對象和代理對象的共同接口;

 

代理角色:代理對象角色內部含有對真實對象的引用,從而可以操作真實對象,同時代理對象提供與真實對象相同的接口以便在任何時刻都能代替真實對象。同時,代理對象可以在執行真實對象操作時,附加其他的操作,相當於對真實對象進行封裝。

 

真實角色:代理角色所代表的真實對象,是我們最終要引用的對象。

 

 

以下是《Java與模式》中的示例爲例:

 

1.1 靜態代理



package staticproxy;

/**
 * 抽象角色
 */
public abstract class Subject {
	public abstract void request();
}



package staticproxy;

/**
 * 真實的角色
 */
public class RealSubject extends Subject {

	@Override
	public void request() {
		// TODO Auto-generated method stub

	}

}



package staticproxy;

/**
 * 靜態代理,對具體真實對象直接引用
 * 代理角色,代理角色需要有對真實角色的引用,
 * 代理做真實角色想做的事情
 */
public class ProxySubject extends Subject {
	
	private RealSubject realSubject = null;
	
	/**
	 * 除了代理真實角色做該做的事情,代理角色也可以提供附加操作,
	 * 如:preRequest()和postRequest()
	 */
	@Override
	public void request() {
		preRequest();  //真實角色操作前的附加操作
		
		if(realSubject == null){
			realSubject =  new RealSubject();
		}
		realSubject.request();
		
		postRequest();  //真實角色操作後的附加操作
	}

	/**
	 *	真實角色操作前的附加操作
	 */
	private void postRequest() {
		// TODO Auto-generated method stub
		
	}

	/**
	 *	真實角色操作後的附加操作
	 */
	private void preRequest() {
		// TODO Auto-generated method stub
		
	}

}



package staticproxy;

/**
 *	客戶端調用 
 */
public class Main {
	public static void main(String[] args) {
		Subject subject = new ProxySubject();
		subject.request();  //代理者代替真實者做事情
	}
}




以賣房(Subject的request方法)爲例,賣房者(RealSubject)想賣房,但是不想搞那麼多麻煩事,不想天天招買房的騷擾,就委託中介(ProxySubject)去幫忙賣房,中介幫忙賣房,當然不會簡單的賣房,可能還需要收取中介費(preRequest()和postRequest()方法)等等附加收費,最終買房者(客戶端Main)買房,找到的是中介,不會與房主接觸,與房主接觸的是中介,買房者最終跟中介買房,完成操作。

 

 

這個是靜態代理。真實角色必須是事先已經存在的,並將其作爲代理對象的內部屬性。但是實際使用時,一個真實角色必須對應一個代理角色,如果大量使用會導致類的急劇膨脹;此外,如果事先並不知道真實角色,該如何使用代理呢?這個問題可以通過Java的動態代理類來解決。

 

動態代理,就相當於代理者不僅僅只是代理一個真實對象,也可以代理很多對象,而且對象是動態指定的。

 

1.2 動態代理

 

Java動態代理類位於Java.lang.reflect包下,一般主要涉及到以下兩個類:

 

(1). Interface InvocationHandler:該接口中僅定義了一個方法Object:invoke(Objectobj,Method method, Object[] args)。在實際使用時,第一個參數obj一般是指代理類,method是被代理的方法,如上例中的request(),args爲該方法的參數數組。這個抽象方法在代理類中動態實現。

 

(2).Proxy:該類即爲動態代理類,作用類似於上例中的ProxySubject,其中主要包含以下內容:

Protected Proxy(InvocationHandler h):構造函數,估計用於給內部的h賦值。

 

Static Class getProxyClass (ClassLoaderloader, Class[] interfaces):獲得一個代理類,其中loader是類裝載器,interfaces是真實類所擁有的全部接口的數組。

 

Static Object newProxyInstance(ClassLoaderloader, Class[] interfaces, InvocationHandler h):返回代理類的一個實例,返回後的代理類可以當作被代理類使用(可使用被代理類的在Subject接口中聲明過的方法)。

 

所謂Dynamic Proxy是這樣一種class:它是在運行時生成的class,在生成它時你必須提供一組interface給它,然後該class就宣稱它實現了這些interface。你當然可以把該class的實例當作這些interface中的任何一個來用。當然啦,這個DynamicProxy其實就是一個Proxy,它不會替你作實質性的工作,在生成它的實例時你必須提供一個handler,由它接管實際的工作。(參見文獻3)

 

在使用動態代理類時,我們必須實現InvocationHandler接口

 

package dynamicproxy;

/**
 * 抽象角色
 * 這裏應改爲接口
 */
public interface Subject {
	void request();
}


package dynamicproxy;

/**
 * 真實的角色
 * 實現接口
 */
public class RealSubject implements Subject {

	@Override
	public void request() {
		// TODO Auto-generated method stub

	}

}



package dynamicproxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

/**
 * 動態代理, 它是在運行時生成的class,在生成它時你必須提供一組interface給它, 然後該class就宣稱它實現了這些interface。
 * 你當然可以把該class的實例當作這些interface中的任何一個來用。 當然啦,這個Dynamic
 * Proxy其實就是一個Proxy,它不會替你作實質性的工作, 在生成它的實例時你必須提供一個handler,由它接管實際的工作。
 */
public class DynamicSubject implements InvocationHandler {

	private Object sub; // 真實對象的引用

	public DynamicSubject(Object sub) {
		this.sub = sub;
	}

	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("before calling " + method); 
        method.invoke(sub,args); 
        System.out.println("after calling " + method); 
        return null; 
	}

}




package dynamicproxy;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

public class Main {
	public static void main(String[] args) throws Throwable {
		RealSubject rs = new RealSubject();
		InvocationHandler handler = new DynamicSubject(rs);
		Class cls = rs.getClass();
		//以下是分解步驟
		/*
		Class c = Proxy.getProxyClass(cls.getClassLoader(), cls.getInterfaces());
		Constructor ct = c.getConstructor(new Class[]{InvocationHandler.class});
		Subject subject =(Subject) ct.newInstance(new Object[]{handler});
		*/
		
		//以下是一次性生成
		Subject subject = (Subject)Proxy.newProxyInstance(cls.getClassLoader(),cls.getInterfaces(), handler);
		subject.request();
	}
}


通過這種方式,被代理的對象(RealSubject)可以在運行時動態改變,需要控制的接口(Subject接口)可以在運行時改變,控制的方式(DynamicSubject類)也可以動態改變,從而實現了非常靈活的動態代理關係

 

1.3.代理模式使用原因和應用方面

 

(1)授權機制不同級別的用戶對同一對象擁有不同的訪問權利,如Jive論壇系統中,就使用Proxy進行授權機制控制,訪問論壇有兩種人:註冊用戶和遊客(未註冊用戶),Jive中就通過類似ForumProxy這樣的代理來控制這兩種用戶對論壇的訪問權限.

 

(2)某個客戶端不能直接操作到某個對象,但又必須和那個對象有所互動.

     舉例兩個具體情況:

     如果那個對象是一個是很大的圖片,需要花費很長時間才能顯示出來,那麼當這個圖片包含在文檔中時,使用編輯器或瀏覽器打開這個文檔,打開文檔必須很迅速,不能等待大圖片處理完成,這時需要做個圖片Proxy來代替真正的圖片.

    

     如果那個對象在Internet的某個遠端服務器上,直接操作這個對象因爲網絡速度原因可能比較慢,那我們可以先用Proxy來代替那個對象.

 

     總之原則是,對於開銷很大的對象,只有在使用它時才創建,這個原則可以爲我們節省很多寶貴的Java內存. 所以,有些人認爲Java耗費資源內存,我以爲這和程序編制思路也有一定的關係.

 

(3)現實中,Proxy應用範圍很廣,現在流行的分佈計算方式RMI和Corba等都是Proxy模式的應用

 

 




發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章