代理模式,顾名思义,就是用一个代理对象,去代替真实对象,完成一些事情。
那么我们来写一个demo。以买房流程为例,我们要找中介、房产公司这种代理,去帮助我们完成买房这件事。
既然代理对象要代替真实对象完成任务,那么它们需要拥有相同的方法。这样调用方才能够通过调用代理对象的方法,进而由代理对象来对真实对象进行调用。
首先要有一个接口
public interface BuyHouse {
void buyHouse();
}
接着,我们有一个房产公司,实现了这个接口,这个是提供买房方法的真实对象。
public class HouseCompany implements BuyHouse {
@Override
public void buyHouse(){
System.out.println("买房成功");
}
}
还要有一个中介,中介实际上就是代理对象
public class Intermediary implements BuyHouse {
private BuyHouse buyhouse;
public Intermediary(BuyHouse buyhouse) {
this.buyhouse = buyhouse;
}
@Override
public void buyHouse(){
System.out.println("售前处理");
buyhouse.buyHouse();
System.out.println("售后处理");
}
}
上面的这种方式,我们称为静态代理。实际上我们还有一种代理方式叫做动态代理,是利用Java的反射技术,在运行的时候为我们动态创建代理对象。
首先,编写动态处理类
public class DynamicProxyHandler implements InvocationHandler{
private Object object;
public DynamicProxyHandler(Object object){
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
System.out.println("售前处理");
Object result = method.invoke(object, args);
System.out.println("售后处理");
return result;
}
}
如何去调用呢?
BuyHouse buyHouse = new HouseCompany();
BuyHouse proxyBuyHouse = (BuyHouse) Proxy.newProxyInstance(BuyHouse.class.getClassLoader(), new Class[] {BuyHouse.class}), new DynamicProxyHandler (buyHouse);
proxyBuyHouse.buyHouse();
Proxy.newProxyInstance()
方法接受三个参数:
ClassLoader loader
指定当前目标对象使用的类加载器,获取加载器的方法是固定的
Class<?>[] interfaces
指定目标对象实现的接口的类型,使用泛型方式确认类型
InvocationHandler
指定动态处理器,执行目标对象的方法时,会触发事件处理器的方法
代理模式的使用情景有如下几个:
1.远程代理。也就是为一个对象在不同的地址空间提供局部代表。这样可以隐藏一个对象存在于不同地址空间的事实。
2.虚拟代理。是根据需要创建开销很大的对象。通过它来存放实例化需要很长时间的真实对象。比如浏览器加载一个有很多图片的网站时,浏览器不会等所有的图片都加载完才打开页面,而是先打开了文字,而那些图片,会慢慢加载出来。那些未加载的图片,就是用虚拟代理代替了真实的图片,代理存储了真实图片的路径和尺寸。
3.安全代理。用来控制真实对象访问时的权限。
4.智能指引。当调用真实对象的时候,代理处理另外一些事。比如加入日志,操作审计,检查锁定等等。
相关demo可以参考我的gitee仓库
https://gitee.com/akitsuki-kouzou/DesignPatternDemo