程序员

动态代理

2017-03-02  本文已影响0人  昵称全尼马被注册了

为什么使用动态代理?


从静态代理说起

什么是代理模式

代理模式分为"静态代理"和"动态代理"。
代理模式主要的功能是控制对对象的访问。比如“同步”,“事务”,“权限”等。

  1. Remember that the main intent of a proxy is to control access to
    the target object, rather than to enhance the functionality of the
    target object
  1. Ways that proxies can provide access control include:
    → Synchronization
    → Authentication
    → Remote Access
    → Lazy instantiation

静态代理

先来看一个简单的接口,一个汽车的接口有启,停,前,退四个方法。

public interface IVehicle {
   public void start();
   public void stop();
   public void forward();
   public void reverse();
 }

给它一个简单的实现类:

public class Car implements IVehicle {
    public void start() {
        System.out.println("Car started");
    }
    public void stop() {
        System.out.println("Car stopped");
    }
    public void forward() {
        System.out.println("Car forwarded");
    }
    public void reverse() {
        System.out.println("Car reversed");
    }
 }

假设有个需求,要记录汽车启,停,前,退方法被调用时的时间。
最low的方法是直接修改Car这个类,但这样的程序毫无扩展性可言,也不符合面向对象的设计规范。

public class Car implements IVehicle {
    public void start() {
        System.out.println("Car started at " + new Date());
        System.out.println("Car started");
    }
    public void stop() {
        System.out.println("Car stopped at " + new Date());
        System.out.println("Car stopped");
    }
    public void forward() {
        System.out.println("Car forwarded at " + new Date());
        System.out.println("Car forwarded");
    }
    public void reverse() {
        System.out.println("Car reversed at " + new Date());
        System.out.println("Car reversed");
    }
 }

记录时间这个逻辑不应该在Car的职责范围类,所以我们引入“静态代理”来解决这个问题。

public class CarProxy implements IVehicle {
    private IVehicle vehicle;
    public CarProxy(IVehicle  vehicle){
        this.vehicle=vehicle;
    }
    public void start() {
        System.out.println("Car started at " + new Date());
        this.vehicle.start();
    }
    public void stop() {
        System.out.println("Car stopped at " + new Date());
        this.vehicle.stop();
    }
    public void forward() {
        System.out.println("Car forwarded at " + new Date());
        this.vehicle.forward();
    }
    public void reverse() {
        System.out.println("Car reversed at " + new Date());
        this.vehicle.reverse();
    }
 }

这样做的确分离了代理逻辑,但同样存在问题:

  1. 哪天IVehicle要加更多的方法进来,CarProxy不也要跟着加代码?
  2. CarProxy里的代理逻辑看起来不重复吗?
  3. 如果其它的类,也需要如此的代理逻辑,重新给他们写个代理类吗?
    </br>
    </br>

动态代理

JDK 动态代理

首先实现ProxyHander,即实现代理的逻辑,也就是记录时间

public class ProxyHander implements InvocationHandler {
    private Object proxiedObject;

    public ProxyHander(Object proxiedObject) {
        this.proxiedObject = proxiedObject;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        
        //代理的逻辑,记录时间
        System.out.println("Car " + method.getName() + " at " + new Date());
        
        //调用被代理类的方法
        return method.invoke(proxiedObject, args);
        
    }
}

使用Proxy.newProxyInstance()去实例化一个代理类

public class Main {
    public static void main(String[] args) {
        
        IVehicle carProxy = (IVehicle)Proxy.newProxyInstance(
                IVehicle.class.getClassLoader(),
                new Class[]{IVehicle.class}, 
                new ProxyHander(new Car())
            );
        
        carProxy.start();
        carProxy.stop();
        carProxy.forward();
        carProxy.reverse();
    }
}

输出

Car start at Thu Mar 02 21:42:49 CST 2017
Car started
Car stop at Thu Mar 02 21:42:49 CST 2017
Car stopped
Car forward at Thu Mar 02 21:42:49 CST 2017
Car forwarded
Car stop at Thu Mar 02 21:42:49 CST 2017
Car stopped

可以看出,实现JDK的动态代理还是比较简单的,相对于静态代理,这样的实现在程序扩展性上会更好。

  1. 如果IVehicle加如了更多的方法,我们不需要做任何修改。
  2. 如果有其它类的对象,也需要类似的代理逻辑,我们可以重用ProxyHander

JDK 动态代理的不足

不足是只有实现了某个接口的类可以使用Java动态代理机制。但是,事实上使用中并不是遇到的所有类都会给你实现一个接口。因此,对于没有实现接口的类,目前无法使用该机制。

cglib

待续...
</br>
</br>


Code:

Sample Code on Github
</br>
</br>


参考:

浅谈Java动态代理
Java动态代理总结
New Tricks with Dynamic Proxies in Java 8
Dynamic Proxies In Java
代理模式及Java实现动态代理

上一篇 下一篇

猜你喜欢

热点阅读