设计模式之代理模式(十七)
现在想买一辆二手车,虽然可以自己找车源,做质量检测一系列的车辆过户流程,这特别浪费时间和精力。于是我想通过中介公司来买车,他们来帮我找想要的车源,帮我办理车辆过户流程。我只需要负责选择喜欢的车,然后付钱就可以啦。
在实际生活中比如房屋中介, 汽车中介,海外代购都是类似于代理模式的方式。
一、代理模式的定义
代理模式就是给一个对象提供一个代理,并由代理对象控制对原对象的引用,在代理模式中,“第三者” 代理主要起到一个中介的作用,它连接客户端和目标对象。
二、这是类图
三、代理模式的结构
-
Subject
抽象角色。声明真实对象和代理对象的共同接口。 -
Proxy
代理角色。代理对象与真实对象实现相同的接口,所以它能够在任何时刻都能够代理真实对象。 -
RealSubject
真实角色。它代表着真实对象,是最终要引用的对象。
四、静态模式与动态模式
我们有多种不同的方式来实现代理,按照代理创建的时期进行分类的话。可以分为静态代理和动态代理。
1、静态代理
由程序员创建或特定工具自动生成源代码,在对其编译。在运行之前,代理类.class 文件已经创建。
- 优点
可以做到符合开闭原则的情况下对目标对象进行功能扩展。
- 缺点
我们为每一个服务都要创建代理类,工作量大,不易管理。同时接口一旦发生变化,类也要相应的修改。
2、动态代理
是在程序运行时通过反射机制动态创建的。
相对于静态代理,动态代理大大减少了我们的开发任务,同时减少了对业务接口的依赖,降低了耦合度。但是缺点是它始终无法摆脱仅支持interface代理的桎梏。
五、情景代码
小明最近要买房子,接下来我们就用代码来实现小明买房子的过程。
1、静态代理
- Subject 抽象接口
public interface Subject {
// 买房子方法
void buyHouse();
}
- RealSubjcet具体实现类
public class RealSubject implements Subject {
@Override3public void buyHouse() {
System.out.println("买房子");
}
}
- Proxy代理类
public class Proxy implements Subject {
Subject subject;
public Proxy (Subject subject) {
this.subject = subject;
}
@Override
public void buyHouse() {
subject.buyHouse();
System.out.println("商量价格");
System.out.println("签合同");
}
}
- 测试类
public class Test {
public static void main(String[] args) {
System.out.println("==非代理模式==");
Subject subject = new RealSubject();
subject.buyHouse();
System.out.println("==静态代理模式==");
Subject proxy = new Proxy(subject);
proxy.buyHouse();
}
}
- 测试结果
==非代理模式==
买房子
==静态代理模式==
买房子
商量价格
签合同
2、动态代理
动态代理是使用反射处理的,如果同学们对反射理解不是很好,可以查看一下反射知识,这里主要介绍设计模式,暂不介绍反射的知识。
- Proxy 代理类
public class DynamicProxy implements InvocationHandler {
Object object;
public DynamicProxy(Object object) {
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = method.invoke(object,args);
System.out.println("商量价格");
System.out.println("签合同");
return result;
}
}
- 测试类
public static void main(String[] args) {
System.out.println("==动态代理模式==");
Subject subject = new RealSubject();
Subject subject1 = (Subject) Proxy.newProxyInstance(Subject.class.getClassLoader(),new Class[] {Subject.class},new DynamicProxy(subject));
subject1.buyHouse();
}
源码GitHub地址:
https://github.com/xiaonongOne/proxy-test/
六、代理模式的优缺点
-
优点
代理模式能够协调调用者和被调用者,在一定程度上降低系统的耦合度。
代理对象可以在客户端和目标对象之间起到中介的作用,这样起到的保护目标对象的作用。 -
缺点
由于客户端和真实类之间增加代理对象,因此有些类型的代理模式可能会造成请求处理速度变慢。
实现代理模式需要额外的代码工作,有些代理模式的实现非常复杂。
七、适用场景
-
远程代理:
为一个对象在不同的地址空间提供局部代表,这样可以隐藏一个对象存在于不同地址空间的事实。 -
虚拟代理:
通过使用过一个小的对象代理一个大对象,这样就可以减少系统的开销。 -
保护代理:
用来控制对真实对象的访问权限。
八、总结
代理模式是通过使用引用代理对象来访问真实对象,在这里代理对象充当用于连接客户端和真实对象的中介者。
代理模式主要用于远程代理、虚拟代理和保护代理。其中保护代理可以进行访问权限控制。