代理模式

代理模式:为其他对象提供一种代理以控制对这个对象的访问。

一、如何理解代理模式的定义

代理在生活中是一种非常常见的现象,比如奔驰、宝马在中国都有代理商,伊利冷饮在各个城市县城也都有代理商,我们买东西都是到代理商那里买的,而不是直接到厂商那里买。

就如代理模式的定义一样代理商的存在就是控制我们消费者对厂商的访问。下面以买车为例,看看如何用代码解释代理

定义一个买车的接口IBuyCar,所有买车都人都需要实现它

interface IBuyCar {
    //付款
    void payment();

    //取车
    void getCar();
}

定义一个汽车代理商CarProxy

class CarProxy implements IBuyCar {

    //在构造方法内传入需要买车的消费者
    private IBuyCar mIBuyCar = null;

    public CarProxy(IBuyCar iBuyCar){
        mIBuyCar = iBuyCar;
        System.out.println("代理商收到购车要求");
    }
    //通过代理商向商家付款
    @Override
    public void payment() {
        mIBuyCar.payment();
    }

    //通过代理商向商家取车
    @Override
    public void getCar() {
        mIBuyCar.getCar();
    }
}

定义一个汽车买家类

class CarBuyer implements IBuyCar {
    private String name;

    public CarBuyer(String name){
        this.name = name;
        System.out.println(name + "准备买辆汽车");
    }

    @Override
    public void payment() {
        System.out.println(name + "向商家付款买车");
    }

    @Override
    public void getCar() {
        System.out.println(name + "取车成功");
    }
}

现在可以去买车了

    public static void main(String[] args){
        //创建一个消费者
        IBuyCar carBuyer = new CarBuyer("二丫");
        //创建一个代理
        IBuyCar carProxy = new CarProxy(carBuyer);
        //付款
        carProxy.payment();
        //取车
        carProxy.getCar();
    }

运行结果:

二丫准备买辆汽车
代理商收到购车要求
二丫向商家付款买车
二丫取车成功

通过汽车代理商,我们向商家付款成功的买了一辆小汽车了。

二、定义通用的代理模式

代理模式为其它对象提供一种代理以控制对这个对象的访问。通用类图如下:

代理模式通用类图

代理模式也叫做委托模式,是一项基本的设计技巧。

  • Subject:抽象主题、接口
  • RealSubject:具体主题角色,也被称作委托角色,被代理角色Proxy
  • Proxy:代理,委托类、代理类

下面来看一下代理模式的通用写法:

创建Subject主题

interface Subject {
    //定义一个方法代表。
    void request();
}

定义RealSubject

class RealSubject implements Subject {
    //处理具体业务逻辑
    @Override
    public void request() {
        System.out.println("真实的请求");
    }
}

定义Proxy

class Proxy implements Subject {
    //要代理的实现类
    Subject mSubject = null;
    //构造方法初始化代理类
    public Proxy(Subject subject){
        mSubject = subject;
    }
    @Override
    public void request() {
        this.before();
        mSubject.request();
        this.after();
    }
    //预处理
    private void before(){
        //do something
        System.out.println("预处理");
    }
    //善后处理
    private void after(){
        //do something
        System.out.println("善后处理");
    }
}

最简单的运行:

   public static void main(String[] args){
        Subject realSubject = new RealSubject();
        Subject proxy = new Proxy(realSubject);
        proxy.request();
    }

执行结果:

预处理
真实的请求
善后处理

三、静态代理

上面第一个买车的例子就是一个简单的静态代理。这里我们只需要在代理类中添加一些方法完善一下就行了。
添加预处理和善后处理

class CarProxy implements IBuyCar {
    //在构造方法内传入需要买车的消费者
    private IBuyCar mIBuyCar = null;
    public CarProxy(IBuyCar iBuyCar){
        mIBuyCar = iBuyCar;
        System.out.println("代理商收到购车要求");
    }
    //通过代理商向商家付款
    @Override
    public void payment() {
        beforePay();
        mIBuyCar.payment();
        afterPay();
    }
    //通过代理商向商家取车
    @Override
    public void getCar() {
        beforeGetCar();
        mIBuyCar.getCar();
        afterGetCar();
    }
    //预处理
    private void beforePay(){
        //do something
        System.out.println("付款之前:需支付8000金融服务费");
    }
    //善后处理
    private void afterPay(){
        //do something
        System.out.println("付款之后:等待取车");
    }
    //预处理
    private void beforeGetCar(){
        //do something
        System.out.println("取车之前:签验收合同");
    }
    //善后处理
    private void afterGetCar(){
        //do something
        System.out.println("取车之后:刚开200米,漏油了!");
    }
}

再次执行结果如下:

二丫准备买辆汽车
代理商收到购车要求
付款之前:需支付8000金融服务费
二丫向商家付款买车
付款之后:等待取车
取车之前:签验收合同
二丫取车成功
取车之后:刚开200米,漏油了!

为什么要专门加上这段预处理还善后处理的代码?

这是静态代理的精髓所在,因为CarBuyer之前是只有支付和取车的功能,而通过代理使得CarBuyer增加了预处理和善后处理的功能,扩展了它的功能

为什么叫静态代理?
因为需要事先创建好代理类,区别于动态代理在需要的时候才动态创建。

静态代理优缺点

  • 优点:做到了在符合开闭原则的情况下对目标对象进行功能扩展。
  • 缺点:代理对象必须提前写出,如果接口层发生了变化,代理对象的代码也要进行维护。

四、动态代理

动态代理也叫JDK代理,在动态代理中我们不需要再创建代理类了,只需要一个动态处理器就可以了。

动态代理之所以又叫JDK代理,因为动态代理使用的是Java JDK提供的代理方法。首先来看Java创建代理对象的方法。

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)

newProxyInstance方法有三个参数:

  • 第一个参数:类加载器
  • 第二个参数:目标对象实现的接口的类型
  • 第三个参数:调用处理器,执行目标对象的方法时,会触发事件处理器的方法

第一个和第二个参数写法都是固定的,第三个参数InvocationHandler是一个接口,负责处理具体的逻辑

下面创建InvocationHandler接口的实现类

class CarHandler implements InvocationHandler {

    //被代理的对象
    private Object target = null;

    public CarHandler(Object target){
        this.target = target;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object result = null;
        if (method.getName().equals("payment")){
            beforePay();
            result = method.invoke(target,args);
            afterPay();
        }else if (method.getName().equals("getCar")){
            beforeGetCar();
            result = method.invoke(target,args);
            afterGetCar();
        }
        return result;
    }

    //预处理
    private void beforePay(){
        //do something
        System.out.println("付款之前:需支付8000金融服务费");
    }

    //善后处理
    private void afterPay(){
        //do something
        System.out.println("付款之后:等待取车");
    }

    //预处理
    private void beforeGetCar(){
        //do something
        System.out.println("取车之前:签验收合同");
    }

    //善后处理
    private void afterGetCar(){
        //do something
        System.out.println("取车之后:刚开200米,漏油了!");
    }
}

实现InvocationHandler接口需要实现它的invoke方法

invoke(Object proxy, Method method, Object[] args)

invoke方法也有三个,第一个proxy是代理类的实例,第二个method是方法的反射包装,第三个是方法的参数。

关于反射有不了解的地方可以看下我的 学习Java反射机制

创建好动态代理的核心接口InvocationHandler 的实现类之后,就可以使用动态代理买车了。买车代码:

    public static void main(String[] args){
        //创建被代理对象
        IBuyCar carBuyer = new CarBuyer("老王");
        //动态生成一个代理
        IBuyCar proxy = (IBuyCar) Proxy.newProxyInstance(CarBuyer.class.getClassLoader(),
                CarBuyer.class.getInterfaces(),new CarHandler(carBuyer));
        proxy.payment();
        proxy.getCar();
    }

运行结果:

老王准备买辆汽车
付款之前:需支付8000金融服务费
老王向商家付款买车
付款之后:等待取车
取车之前:签验收合同
老王取车成功
取车之后:刚开200米,漏油了!

使用动态代理我们只需要实现核心接口 InvocationHandler ,代理类由代码运行时JDK创建,不需要我们手动创建。

参考:《设计模式之禅》

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

推荐阅读更多精彩内容