一、简介
代理模式的定义:由于某些原因需要给某对象提供一个代理以控制对该对象的访问。这时,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介。
二、优缺点
优点:
- 代理模式在客户端与目标对象之间起到一个中介作用和保护目标对象的作用;
- 代理对象可以扩展目标对象的功能;
- 代理模式能将客户端与目标对象分离,在一定程度上降低了系统的耦合度,增加了程序的可扩展性
缺点:
- 代理模式会造成系统设计中类的数量增加
- 在客户端和目标对象之间增加一个代理对象,会造成请求处理速度变慢;
- 增加了系统的复杂度;
三、应用场景
- 远程代理,这种方式通常是为了隐藏目标对象存在于不同地址空间的事实,方便客户端访问。例如,用户申请某些网盘空间时,会在用户的文件系统中建立一个虚拟的硬盘,用户访问虚拟硬盘时实际访问的是网盘空间。
- 虚拟代理,这种方式通常用于要创建的目标对象开销很大时。例如,下载一幅很大的图像需要很长时间,因某种计算比较复杂而短时间无法完成,这时可以先用小比例的虚拟代理替换真实的对象,消除用户对服务器慢的感觉。
- 安全代理,这种方式通常用于控制不同种类客户对真实对象的访问权限。
- 智能指引,主要用于调用目标对象时,代理附加一些额外的处理功能。例如,增加计算真实对象的引用次数的功能,这样当该对象没有被引用时,就可以自动释放它。
- 延迟加载,指为了提高系统的性能,延迟对目标的加载。例如,Hibernate 中就存在属性的延迟加载和关联表的延时加载。
四、模式结构
1. 主要角色
- 抽象主题(Subject)类:通过接口或抽象类声明真实主题和代理对象实现的业务方法。
- 具体主题(Real Subject)类:实现了抽象主题中的具体业务,是代理对象所代表的真实对象,是最终要引用的对象。
- 代理(Proxy)类:提供了与真实主题相同的接口,其内部含有对真实主题的引用,它可以访问、控制或扩展真实主题的功能。
2. UML类图
五、代码实现
1) 抽象主题类
public interface Travel {
/**
* 买票
*/
void buyTicket();
}
2) 具体主题类
public class TravelPerson implements Travel {
@Override
public void buyTicket() {
System.out.println("早上9点出发,北京——>上海.");
}
}
3) 静态代理及测试
public class StaticTravelProxy implements Travel {
private TravelPerson travelPerson = null;
public StaticTravelProxy(TravelPerson travelPerson){
this.travelPerson = travelPerson;
}
@Override
public void buyTicket() {
this.preBuyTicket();
this.travelPerson.buyTicket();
this.endBuyTicket();
}
public void preBuyTicket(){
System.out.println("买票之前的操作。");
}
public void endBuyTicket(){
System.out.println("买票后的操作。");
}
}
@Test
public void test(){
StaticTravelProxy staticTravelProxy = new StaticTravelProxy(new TravelPerson());
staticTravelProxy.buyTicket();
}
4) JDK动态代理类及测试
public class JDKTravelProxy implements InvocationHandler {
// 被代理对象
private Object traget = null;
/**
* 获取动态代理对象
* @param traget
* @return
*/
public Object getInstance(Object traget){
this.traget = traget;
Class<?> clazz = this.traget.getClass();
/**
* Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)生成一个代理对象
* ClassLoader loader:代理对象的类加载器,一般使用被代理对象的类加载器
* Class<?>[] interfaces:代理对象实现的接口,一般使用被代理对象实现的接口
* InvocationHandler h:执行处理类,一般为当前类
*/
Object proxy = Proxy.newProxyInstance(clazz.getClassLoader(), clazz.getInterfaces(), this);
return proxy;
}
/**
* 调用代理类中的任何方法,此方法都会执行
* @param proxy 代理对象
* @param method 当前执行的方法
* @param args 当前执行的方法参数
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
this.preBuyTicket();
// 调用代理对象方法
Object obj = method.invoke(this.traget, args);
this.endBuyTicket();
return obj;
}
public void preBuyTicket(){
System.out.println("买票之前的操作。");
}
public void endBuyTicket(){
System.out.println("买票后的操作。");
}
}
@Test
public void test() {
Travel travel = (Travel) new JDKTravelProxy().getInstance(new TravelPerson());
// 通过调用JDKTravelProxy的invoke方法来实现代理
travel.buyTicket();
}