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