动态代理

为什么使用动态代理?

  • 相对于静态代理,减少了一些代码量。
  • 相对于静态代理,程序扩展性更好。
  • 动态代理中,代理类的逻辑,与被代理类的逻辑,完全分离开来,耦合度降低
  • 动态代理的优势就是实现无侵入式的代码扩展
  • 动态代理实现了在原始类和接口未知的时候,就确定代理的代理行为,当代理类和原始类脱离直接联系后,就可以灵活的重用到不同的应用场景中
    </br>
    </br>

从静态代理说起

什么是代理模式

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

  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实现动态代理

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349

推荐阅读更多精彩内容

  • 前言 Java动态代理通过反射的机制实现在运行时,基于传入的指定一组接口及委托类对象,动态的产生代理类,代理类负责...
    Justlearn阅读 991评论 1 10
  • 代理模式 定义 为实际对象提供一个代理,以控制对实际对象的访问。代理类负责为委托类预处理消息,过滤消息并转发消息,...
    CoderBao阅读 1,220评论 0 0
  • Java代理和动态代理机制分析和应用 概述 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个...
    丸_子阅读 3,016评论 6 57
  • 在北京的第二天下午把自己安顿在了青社,还经历了一场冰雹,逃离北京的念头终于消失了。晚上出去遛弯自己竟然摸去了银泰百...
    夏天的风xy阅读 205评论 0 0
  • 昨天对毕桢发脾气了,因为他的饮食无节制,导致呕吐,吐在卧室——地板,床垫,被子上都是,还吐在了下午老师要求...
    温暖甜阅读 490评论 0 3