在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。
在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。
所谓的代理者是指一个类别可以作为其它东西的接口。代理者可以作任何东西的接口:网上连接、存储器中的大对象、文件或其它
昂贵或无法复制的资源。
著名的代理模式例子为引用计数(英语:reference counting)指针对象。
当一个复杂对象的多份副本须存在时,代理模式可以结合享元模式以减少存储器用量。典型作法是创建一个复杂对象及多个代理
者,每个代理者会引用到原本的复杂对象。而作用在代理者的运算会转送到原本对象。一旦所有的代理者都不存在时,复杂对象会
被移除
组成
抽象角色:通过接口或抽象类声明真实角色实现的业务方法。
代理角色:实现抽象角色,是真实角色的代理,通过真实角色的业务逻辑方法来实现抽象方法,并可以附加自己的操作。
真实角色:实现抽象角色,定义真实角色所要实现的业务逻辑,供代理角色调用。 [2]
优点
(1).职责清晰
真实的角色就是实现实际的业务逻辑,不用关心其他非本职责的事务,通过后期的代理完成一件完成事务,附带的结果就是编程
简洁清晰。
(2).代理对象可以在客户端和目标对象之间起到中介的作用,这样起到了中介的作用和保护了目标对象的作用。
模式结构
一个是真正的你要访问的对象(目标类),一个是代理对象,真正对象与代理对象实现同一个接口,先访问代理类再访问真正要访问的
对象。
代理模式分为静态代理、动态代理。
静态代理是由程序员创建或工具生成代理类的源码,再编译代理类。所谓静态也就是在程序运行前就已经存在代理类的字节码文
件,代理类和委托类的关系在运行前就确定了。
动态代理是在实现阶段不用关心代理类,而在运行阶段才指定哪一个对象。
使用场景
按职责来划分,通常有以下使用场景: 1、远程代理。 2、虚拟代理。 3、Copy-on-Write 代理。 4、保护(Protect or
Access)代理。 5、Cache代理。 6、防火墙(Firewall)代理。 7、同步化(Synchronization)代理。 8、智能引用(Smart
Reference)代理。
实现
我们将创建一个 Press接口和实现了 Press接口的实体类Bookstore。ProxyISellBook 是一个代理类,减少 Bookstore对象加载的
内存占用。Demo,我们的演示类使用 ProxyISellBook 来获取要加载的 Press对象,并按照需求进行显示。
Demo
public interface Press {
public void sell();
}
public class Bookstore implements Press{
@Override
public void sell() {
System.out.println("卖书");
}
}
public class ProxySellBook implements Press{
private Bookstore bookstore;
@Override
public void sell() {
dazhe();
if(bookstore == null){
bookstore=new Bookstore();
}
bookstore.sell();
zengjf();
}
public void dazhe(){
System.out.println("打五折");
}
public void zengjf(){
System.out.println("赠送积分100");
}
}
public class Demo {
public static void main(String[] args) {
ProxySellBook ps=new ProxySellBook();
ps.sell();
}
}
动态代理
JDK 自带的动态代理
- java.lang.reflect.Proxy:生成动态代理类和对象;
- java.lang.reflect.InvocationHandler(处理器接口):可以通过invoke方法实现
对真实角色的代理访问。
每次通过 Proxy 生成的代理类对象都要指定对应的处理器对象。
代码
public interface Subject {
public void sellBooks();
}
public class RealSubject implements Subject {
@Override
public final void sellBooks() {
System.out.println("卖书");
}
}
public class MyInvocaticonHandler implements InvocationHandler {
RealSubject realSubject;
public void setRealSubject(RealSubject realSubject) {
this.realSubject = realSubject;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result=null;
dazhe();
result=method.invoke(realSubject, args);
zengjf();
return result;
}
public void dazhe(){
System.out.println("打五折");
}
public void zengjf(){
System.out.println("赠送积分100");
}
}
public class DTDemo {
public static void main(String[] args) {
RealSubject pr=new RealSubject();
MyInvocaticonHandler mi=new MyInvocaticonHandler();
mi.setRealSubject(pr);
//代理对象
Subject pre=(Subject) Proxy.newProxyInstance(RealSubject.class.getClassLoader(),pr.getClass().getInterfaces(),mi);
pre.sellBooks();
}
}
输出结果: