Java基础

设计模式3.5 代理模式

2018-11-30  本文已影响1人  卢卡斯哔哔哔

点击进入我的博客

代理模式给某一个对象提供一个代理对象,并由代理对象控制对原对象对引用

3.5.1 代理模式结构

代理模式
public class Test {
    public static void main(String[] args) {
        Subject subject = new RealSubject();
        Subject proxy = new ProxySubject(subject);
        proxy.request(); // 此处通过代理类来执行
    }
}

interface Subject {
    void request();
}

class RealSubject implements Subject {
    @Override
    public void request() {
        System.out.println("RealSubject");
    }
}

class ProxySubject implements Subject {
    private Subject subject;

    public ProxySubject(Subject subject) {
        this.subject = subject;
    }

    @Override
    public void request() {
        System.out.println("ProxySubject");
    }
}

3.5.2 动态代理

自从JDK 1.3以后,Java在java.lang.reflect库中提供了一下三个类直接支持代理模式:ProxyInvocationHanderMethod

动态代理步骤
  1. 创建一个真实对象
  2. 创建一个与真实对象有关的调用处理器对象InvocationHandler
  3. 创建代理,把调用处理器和要代理的类联系起来Proxy.newInstance()
  4. 在调用处理对象的invoke()方法中执行相应操作
public class Test {
    public static void main(String[] args) {
        // 创建要被代理的实例对象
        Subject subject = new RealSubject();
        // 创建一个与被代理实例对象有关的InvocationHandler
        InvocationHandler handler = new ProxySubject(subject);
        // 创建一个代理对象来代理subject,被代理的对象subject的每个方法执行都会调用代理对象proxySubject的invoke方法
        Subject proxySubject = (Subject) Proxy.newProxyInstance(Subject.class.getClassLoader(), new Class[]{Subject.class}, handler);
        // 代理对象执行
        proxySubject.request();
    }
}

interface Subject {
    void request();
}

class RealSubject implements Subject {
    @Override
    public void request() {
        System.out.println("RealSubject");
    }
}

class ProxySubject implements InvocationHandler {
    private Subject subject;

    public ProxySubject(Subject subject) {
        this.subject = subject;
    }

    /**
     * @param proxy 要代理的
     * @param method
     * @param args
     * @return
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("Before Proxy");
        Object obj = method.invoke(subject, args);
        System.out.println("After Proxy");
        return obj;
    }
}

3.5.3 细节

优点
  1. 代理类和真实类分离,职责清晰。
  2. 在不改变真是累代码的基础上扩展了功能。
缺点
  1. 由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。
  2. 实现代理模式需要额外的工作,有些代理模式的实现非常复杂。
和适配器模式的关系

适配器模式的用意是改变所考虑对象的接口,而代理模式不能改变。

和装饰模式
虚拟代理
上一篇 下一篇

猜你喜欢

热点阅读