目录
前言:
在有些情况下,一个客户不能或者不想直接访问另一个对象,这时需要找一个中介帮忙完成某项任务,这个中介就是代理对象。例如,购买火车票不一定要去火车站买,可以通过 12306 网站或者去火车票代售点买。又如找女朋友、找保姆、找工作等都可以通过找中介完成。
在软件设计中,使用代理模式的例子也很多,例如,要访问的远程对象比较大(如视频或大图像等),其下载要花很多时间。还有因为安全原因需要屏蔽客户端直接访问真实对象,如某单位的内部数据库等。
定义与特点:
代理模式的定义:由于某些原因需要给某对象提供一个代理以控制对该对象的访问。这时,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介。
代理模式的主要特点有:
-
代理模式会造成系统设计中类的数量增加
-
在客户端和目标对象之间增加一个代理对象,会造成请求处理速度变慢;
-
增加了系统的复杂度;
结构与实现:
举例:一个房东需要中介来代理他出租房子。
结构图:
静态代理模式:
//接口——租房
public interface Rent {
void rent();
}
//真实角色——房东
public class Host implements Rent{
@Override
public void rent() {
System.out.println("房东要出租房子");
}
}
//代理角色——中介
public class Proxy implements Rent {
private Host host;
public Proxy(Host host) {
this.host = host;
}
@Override
public void rent() {
host.rent();
fare();
seeHouse();
}
public static void fare(){
System.out.println("中介收费");
}
public static void seeHouse(){
System.out.println("中介带你看房");
}
}
public class Test {
public static void main(String[] args) {
//静态代理模式
//房东出租房子
Host host = new Host();
//中介代理房东出租
Proxy proxy = new Proxy(host);
//找中介租房
proxy.rent();
}
结果:
房东要出租房子
中介收费
中介带你看房
动态代理模式:
//真实角色——房东
public class Host implements Rent{
@Override
public void rent() {
System.out.println("房东要出租房子");
}
}
//接口——租房
public interface Rent {
void rent();
}
public class ProxyInvocationHandler implements InvocationHandler {
//被代理的接口
private Object target;
public void setTarget(Object target){
this.target = target;
}
//生成得到代理
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
}
//处理代码实例,并返回结果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = method.invoke(target, args);
return result;
}
}
public class Test {
public static void main(String[] args) {
//动态代理模式
//真实角色
Host hostdong = new Host();
//代理角色,不存在
ProxyInvocationHandler pih = new ProxyInvocationHandler();
//设置要代理的对象
pih.setTarget(hostdong);
//动态生辰代理类
Rent proxy1 = (Rent)pih.getProxy();
proxy1.rent();
}
}
结果:
房东要出租房子