代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问。
某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。
代理模式一般涉及到的角色有:
1、抽象角色:声明真实对象和代理对象的共同接口。
2、代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其它的操作,相当于对真实对象进行封装。
3、真实对象:代理角色所代表的真实对象,是我们最终要引用的对象。
例子:
1、抽象角色:
public abstract class Subject {
public abstract void request();
}
2、真实角色:
public class RealSubject extends Subject {
public void request() {
System.out.println("From real subject!");
}
}
3、代理角色:
public class ProxySubject extends Subject {
private RealSubject realSubject;
public void request() {
this.preRequest();
if(null==realSubject){
realSubject = new RealSubject();
}
realSubject.request();
this.postRequest();
}
private void preRequest(){
System.out.println("pre request");
}
private void postRequest(){
System.out.println("pos request");
}
}
4、客户端:
public class Client {
public static void main(String[] args) {
Subject subject = new ProxySubject();
subject.request();
}
}
如果按照上述的方法使用代理模式,真实角色必须是事先已经存在的,并将其作为代理对象的内部属性。但实际中,一个真实角色必须对应一个代理角色,如果大量使用会导致类的急剧膨胀,此外,如果事先不知道真实角色,可以用动态代理。
动态代理类,Java动态代理类位于java.lang.reflect包下,一般涉及到以下两个类:
1、Interface InvocationHandle:该接口中仅定我了一个方法: public Object invoke(Object obj, Method method, Object[] args)
在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,args是该方法的参数数组。这个抽象方法在代理类中动态实现。
2、Proxy:该类即为动态代理类
protected Proxy(InvocationHandler h):构造函数,用于给内部的h赋值。
static Class getProxy(ClassLoader loader, Class[] interface):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。
static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类命名用。
所谓Dynamic Proxy是这样一个class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然,这个Dynamic Proxy其实就一个Proxy,它不会替你做实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。