为其他对象提供一个代理以控制对这个对象的访问,在不改变目标对象基础上添加额外的功能。
如何理解代理模式
- 两个主体:代理对象和被代理对象
- 对于被代理对象,事情必须要去做吗,但是自己不想去做或是没有条件去做,需要代理对象代替被代理对象去做。
- 代理对象能够获取被代理对象的资料信息。 代理层面(获得被代理对象的引用)
生活案例
中介:通常买二手车的时候,会去网上找车源,对车进行质量检测,以及各种过户手续的办理,自己可能不想做这些事情,所以可以找第三方的中介公司来完成这些事情,我只负责把我想要的车辆的信息(价位、车辆新旧程度、品牌)反馈给中介,中介把所有的办好就只管我来签字验收就可以了。
黄牛:当春运火车票比较紧张的时候,我抢不到票,且自己也不想去抢,则可以通过黄牛去买,我只管把我的车票信息给他(初始地、目的地、出发时间、车次),由此不用抢票,也可以买到相应的车票,抢票的过程由黄牛去做。
媒人:平时自己没有时间去交女朋友,则需要媒人去介绍,我们只管把我们想要的女朋友的类型给媒人,媒人则帮我们去寻找合适的对象。
需要类图
可以是抽象类也可以是接口
被代理的对象,业务逻辑的执行者
代理类,负责对真实角色的应用,把所有抽象主题定义的方法限制委托给真实角色实现,并在真实角色处理完毕前后做相应工作
几种代理实现形式
代理有两种主要形式
- 静态代理
编译前,代码已经创建好,程序运行时,class文件已经存在
- 动态代理
程序运行时通过反射生成代理类
静态代理
- 代理类与目标类实现相同的接口
- 代理类持有目标类的引用,可以控制目标类方法的访问
普通代理
1 2 3 4
| public interface Subject {
void request(); }
|
1 2 3 4 5 6
| public class RealSubject implements Subject { @Override public void request() { System.out.println("start to request"); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| public class Proxy implements Subject{ private Subject subject;
public Proxy(Subject subject){ this.subject = subject; }
@Override public void request() { this.before(); subject.request(); this.after(); }
private void before(){ System.out.println("前置处理。。。。"); }
private void after(){ System.out.println("后置处理。。。。"); } }
|
强制代理
- 只能通过真实角色角色制定的代理类才能访问
- 不允许直接访问真实角色
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50
| public interface Subject { void request();
Subject getProxy(); } ```java public class SubjectProxy implements Subject{ private Subject subject;
public SubjectProxy(Subject subject) { this.subject = subject; }
@Override public void request() { this.subject.request(); }
@Override public Subject getProxy() { return this; } } ```java public class RealSubject implements Subject{ private Subject proxy = null; @Override public void request() { if(this.isProxy()){ System.out.println("start to request"); }else { System.out.println("请使用指定的代理进行访问"); }
}
@Override public Subject getProxy() { this.proxy = new SubjectProxy(this); return this.proxy; }
private boolean isProxy(){ if(this.proxy == null){ return false; }else { return true; } } }
|
1 2 3 4 5 6 7
| public class Client { public static void main(String[] args) { RealSubject subject = new RealSubject(); Subject proxy = subject.getProxy(); proxy.request(); } }
|
静态代理局限性
- 不够灵活,每个代理只为一个接口服务,会造成大量代码重复
- 静态代理类职位特定的接口服务,如果有多个接口,需要生成多个代理对象,不利于系统维护
动态代理
1 2 3
| public interface CustomerService { void buyCar(); }
|
1 2 3 4 5 6
| public class CustomerServiceImpl implements CustomerService{ @Override public void buyCar() { System.out.println("决定买这个车了"); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| public class CarAgency implements InvocationHandler{ private Object target;
public Object getInstance(Object target){ this.target = target; Class clazz = target.getClass(); return Proxy.newProxyInstance(clazz.getClassLoader(),target.getClass().getInterfaces(),this); }
@Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("办理车辆质量检查报告"); method.invoke(target,args); System.out.println("办理车辆二手车过户"); return null; } }
|
动态代理源码解析
- 代理模式的意义(如何案例理解)
- 三种代理方式
- 动态代理实现原理
- 动态代理的实际运用(RPC、Spring等 RMI 收集相关的应用)