代理模式、动态代理及其应用

314 查看

代理模式是二十多种设计模式中的一个,属于比较常用的设计模式。本质上就是用来委托我们生成的代理类去完成一些额外的功能,这样能够达到解耦、封装的目的。
通常可以用在RPC、AOP中。比如在RPC中,当我们调用远程方法时,需要委托代理类帮助我们去通过网络连接远程的服务提供者,帮助我们将消息编码发送给服务端,帮我们接受服务端发来的结果。

静态代理

在代理模式中,通常指的时静态代理。

这是静态代理的UML图。代理的思想是:代理类ProxySubject拥有实际类RealSubject的相同方法doSomething(实现相同的一个接口),同时代理类内聚了实际类(即传入实际类的一个引用),在代理类的doSomething方法中,通过实际类的引用调用实际类的doSomething方法,并在调用前后加入需要代理实现的额外功能。这样我们就可以通过调用代理类的相同方法来达到我们的目的。
举个例子:

    interface Subject{
        void doSomething();
    }
class RealSubject implements Subject{
    void doSomething(){
        System.out.println("doSomething");
    }
}
class ProxySubject implements Subject{
    private Subject target;
    public ProxySubject(Subject target){
        this.target = target;
    }
    void doSomething(){
        //do before
        System.out.println("do before");
        
        target.doSomething();//调用realSubject的doSomething方法
        
        //do after
        System.out.println("do after");
    }
}
class ProxyTest{
    public static void main(String args[]){
        //生成代理类,并将实际类传入
        ProxySubject proxy = new ProxySubject(new RealSubject());
        proxy.doSomething();
    }
}

以上,通过代理类执行实际类相同的方法,我们可以运行额外的功能。这里不仅会运行实际类doSomething方法里面的打印“doSomething”,还会在之前打印“do before",之后打印”do after",这两个即为额外的功能。

动态代理

上面的静态代理其实存在一个问题,一个实际类对应一个代理类,而很多时候需要代理实现的额外功能是相同的,比如我们要在A类的某个方法调用前打印“before”,B类的某个方法调用前打印“before”,C类也有这个需求,如果用静态代理,我们需要写A类的代理,B类的代理,C类的代理。。。这样岂不是要累死。
java当中的动态代理就是为了这个解决这个问题。其实思路也很简单,不用去管A类、B类、C类还是什么其他类,统一用反射调用实际类的方法。
java提供了动态代理类Proxy,使用其中的静态方法Proxy.newProxyInstance(ClassLoader loader,Class<?>[] interfaces, InvocationHandler h)来动态生成代理类。interfaces为实际类的接口,对应上面图中的Subject,loader为接口的Classloader,InvocationHandler接口有一个invoke方法需要自己实现,我们委托代理类实现的额外功能便放在该方法里。需要注意的是由此看出java的动态代理需要有接口才能使用。如果不用接口实现动态代理只能求助于cglib这类字节码增强框架。
下面是动态代理的一般实现方法:

public class InvokeProxy implements InvocationHandler{
    private Object target;
    
    public Object bind(Object target){
        this.target = target;
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), 
                target.getClass().getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        
        System.out.println("do something before");
        method.invoke(target, args);
        System.out.println("do something after");
        
        return null;

        /*
        如果反射调用的方法有返回值:
        Object result;
        result = method.invoke(target, args);
        return result;
        */
    }

}

同样有接口Subject:

 interface Subject{
        void doSomething();
    }

假设类A实现了Subject接口,可以根据如下代码生成A的代理

public class ProxyTest {

    public static void main(String[] args) {
        Subject proxyA = (Subject)new InvokeProxy().bind(new A());
        proxyA.doSomething();
    }

}

如果还有类B实现了Subject接口,我们只需更改上面的代码即可,而不用像静态代理那样重复的编写代码。这就是动态代理相对于静态代理的意义。