美文网首页Java 杂谈Java设计模式
设计模式之代理(Proxy)

设计模式之代理(Proxy)

作者: ikonan | 来源:发表于2018-07-30 10:45 被阅读3次

介绍

代理模式是一种结构型设计模式,它可以为其他对象提供一种代理以控制对这个对象的访问。

所谓代理,是指具有与被代理对象相同的接口的类,客户端必须通过代理与被代理的目标类进行交互,而代理一般在交互的过程中(交互前后),进行某些特定的处理。

UML类图

代理UML图

代理模式中的角色

  • 抽象对象角色:声明了目标类及代理类对象的共同接口,这样在任何可以使用目标对象的地方都可以使用代理对象。
  • 目标对象角色:定义了代理对象所代表的目标对象。
  • 代理对象角色:代理对象内部含有目标对象的引用,从而可以在任何时候操作目标对象;代理对象和目标对象具有统一的接口,以便可以再任何时候替代目标对象。代理对象通常在客户端调用传递给目标对象之前或者之后,执行某些操作,而非单纯的将调用传递给目标对象。

代码示列(静态代理)

首先定义AbstractObject类,在其中定义代理类和目标类共有的接口operation()

public interface AbstractObject {
    void operation();
}

下面定义目标实现类:

public class RealObject implements AbstractObject {

    @Override
    public void operation() {
        System.out.println("do operation...");
    }
}

下面是代理类:

public class ProxyObject implements AbstractObject{

    //对目标类的引用
    private RealObject realObject;

    public ProxyObject(RealObject realObject) {
        this.realObject = realObject;
    }

    @Override
    public void operation() {
        System.out.println("do something before real operation...");
        if (realObject == null) {
            realObject = new RealObject();
        }
        realObject.operation();
        System.out.println("do something after real operation...");
    }
}

下面是测试类:

public class ProxyTest {

    public static void main(String[] args) {
        AbstractObject proxy = new ProxyObject(new RealObject());
        proxy.operation();
    }
}

运行结果:
do something before real operation...
do operation...
do something after real operation...

从这个例子可以看出,代理对象将客户端的调用委派给了目标对象,在调用目标对象之前及之后都可以执行某些特定的操作。

动态代理

介绍:动态代理有别于静态代理,是根据代理的对象,动态创建代理类。这样,就可以避免静态代理中代理类接口过多的问题。动态代理是实现方式,是通过反射来实现的,借助Java自带的java.lang.reflect.Proxy,通过固定的规则生成。

代码示列

其他部分和上面一样,以下示代理对象的代码:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class ProxyHandler implements InvocationHandler {

    Object obj = null;

    public Object newProxyInstance(Object realObj) {
        this.obj = realObj;
        Class<?> classType = this.obj.getClass();
        return Proxy.newProxyInstance(classType.getClassLoader(),
                classType.getInterfaces(),this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object result = method.invoke(obj,args);
        return result;
    }
}

测试代码:

public class ProxyTest {

    public static void main(String[] args) {
        AbstractObject proxy = (AbstractObject) new ProxyHandler()
                .newProxyInstance(new RealObject());
        proxy.operation();
    }
}

运行结果:
do operation...

代理模式的使用场景

如果已有的方法在使用的时候需要对原有的方法进行改进,此时有两种办法:

  • 修改原有的方法来适应。显然这违反了“对扩展开放,对修改关闭”的原则。
  • 采用一个代理类调用原来的方法,且对产生的结果进行控制。这就是代理模式了。

使用代理模式可以将功能划分的更加清晰,有助于后期的维护。

相关文章

网友评论

    本文标题:设计模式之代理(Proxy)

    本文链接:https://www.haomeiwen.com/subject/pyvxvftx.html