关闭

设计模式-代理模式

485人阅读 评论(0) 收藏 举报

设计模式中定义: 为其他对象提供一种代理以控制对这个对象的访问.(在出发点到目的地之间有一道中间层,意为代理.)

代理模式涉及的角色: 
1:抽象主题角色.声明了代理主题和真实主题的公共接口,使任何需要真实主题的地方都能用代理主题代替. 

2:代理主题角色.含有真实主题的引用,从而可以在任何时候操作真实主题,代理主题功过提供和真实主题相同的接口,使它可以随时代替真实主题.代理主题通过持有真实主题的引用,不但可以控制真实主题的创建或删除,可以在真实主题被调用前进行拦截,或在调用后进行某些操作. 

3:真实代理对象.定义了代理角色所代表的具体对象. 



以买票为例:
1.先定义一个公共的买票接口
package com.db.proxy;

public interface BuyInterface {
	public void buy();
}
2.定义一个真正买票的类
package com.db.proxy;

public class BuyTicket implements BuyInterface {

	@Override
	public void buy() {
		System.out.println("i am buying a ticket.");
	}

}
3.定义一个代理来代替买票
package com.db.proxy;

public class ProxyBuyTicket implements BuyInterface {

	private BuyTicket bt;
	
	ProxyBuyTicket(BuyTicket bt){
		this.bt = bt;
	}
	
	@Override
	public void buy() {
		preRequest(); 
		System.out.println("i am using proxy to buy a ticket");
		bt.buy();
		postRequest();
	}
	
	private void preRequest() {
		// something you want to do before requesting
	}

	private void postRequest() {
		// something you want to do after requesting
	}

}
测试代码:
package com.db.proxy;

public class ProxyTest {
	
	public static void main(String[] args) {
		BuyInterface bi = new ProxyBuyTicket(new BuyTicket());
		bi.buy();
	}
}
结果:
i am using proxy to buy a ticket
i am buying a ticket.

以上是最简单的代理实现,利用java的反射机制,可以实现动态代理。
java主要是通过Proxy类和InvocationHandler接口来给实现对代理模式的支持的。

定义一个动态代理类:
package com.db.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class DynamicProxyBuyTicket implements InvocationHandler {
	
	private Object toProxyObj;
	
	DynamicProxyBuyTicket(Object obj){
		this.toProxyObj = obj;
	}
	
	public static Object getProxyObject(Object obj){
		@SuppressWarnings("rawtypes")
		Class cls = obj.getClass();
		return Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(),new DynamicProxyBuyTicket(obj));
	}
	 
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println("pre invoke method:" + method); 
		if(args!=null){
			//list args
			for(int i=0;i<args.length;i++){
				System.out.println(args[i]);
			}
		}
		//call real obj's method
		method.invoke(toProxyObj, args);
		System.out.println("post invoke method:" + method);  
		return null;
	}
}
测试代码:
package com.db.proxy;

public class DynamicProxyTest {
	
	public static void main(String[] args) {
		BuyInterface bi = (BuyInterface)DynamicProxyBuyTicket.getProxyObject(new BuyTicket());
		bi.buy();
	}
}
结果:
pre invoke method:public abstract void com.db.proxy.BuyInterface.buy()
i am buying a ticket.
post invoke method:public abstract void com.db.proxy.BuyInterface.buy()


关于java的动态代理:

Java动态代理类位于Java.lang.reflect包下,一般主要涉及到以下两个类:

(1). Interface InvocationHandler:该接口中仅定义了一个方法Object:invoke(Object obj,Method method, Object[] args)。在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的request(),args为该方法的参数数组。这个抽象方法在代理类中动态实现。


(2).Proxy:该类即为动态代理类,作用类似于上例中的ProxySubject,其中主要包含以下内容:

Protected Proxy(InvocationHandler h):构造函数,估计用于给内部的h赋值。

Static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。

Static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类的在Subject接口中声明过的方法)。

所谓Dynamic Proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然啦,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。(Forest Hou,Dynamic Proxy  Java RMI 中的应用》)







0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:200280次
    • 积分:2261
    • 等级:
    • 排名:第16620名
    • 原创:42篇
    • 转载:22篇
    • 译文:0篇
    • 评论:50条
    文章分类
    最新评论