android设计模式之代理模式

静态代理

基本概念

目标对象为其对象提供一种代理对象,其他对象通过代理对象来控制对目标对象的访问。

角色划分

Proxy(美['prɑksi]):代理对象,其他对象直接控制的对象。
Subject([ˈsʌbdʒɪkt]):目标接口,目标对象的抽象。
RealSubject:具体目标对象,目标接口的实现,即真正控制的对象。

简单理解:
当其他对象对某个目标对象进行操作且需要对目标对象进行更改时,若目标对象被多个对象引用,或目标对象不可以更改,此时使用代理模式,通过对代理对象的修改,而完成最终的业务需求。

来源于生活的案例

  • 案例描述: 我通过代购,买了一台iphone X。

我想要买一台港版的iphone X,但是我在大陆,不能直接购买港版的iphone X。于是,我找到了代购公司,代购公司帮我买一个iphoneX。

目标接口:买手机动作,行为抽象
目标对象:我->买手机的人->买手机的行为
代理对象:代购的人->同学或者朋友->代理买手机的行为
其他对象:调用购买手机的对象
  • 案例理解: 我为什么是目标对象?

    在考虑整个案例之后,大多数人一定会主观认为目标对象应该是手机,但是将每个对象进行行为划分,即可理解为目标对象是我,即我买手机的这个行为。
    整个案例的核心,是我想要买手机,代购的核心是帮我买手机,而手机却是具有其他行为的,与买手机行为无关的对象。
    所以,以买手机的行为方向,进行思考,可以得出以下结论:

目标接口:买手机动作,行为抽象
目标对象:我->买手机的人->买手机的行为
代理对象:代购的人->同学或者朋友->代理买手机的行为
其他对象:调用购买手机的对象
  • 案例代码实现: 代码实现三步走;

1.将行为抽象,实现目标接口

 目标接口:IShopPhone(购物->手机)

2.实现行为,完成购买动作

目标对象:ShopPhone
特点:实现目标接口


3.实现代理对象,改变由代理对象完成。

代理对象:WJProxy
        特点一:实现目标接口(可有可无)
        特点二:持有目标对象的引用(必需)
源码展示
/**
 * 目标接口,抽象购买行为
 */
public interface IShopPhone {
    /**
     * 购买手机
     * @param str
     */
    void shopPhone(String str);
}

/**
 * 目标接口的实现类,实现具体的行为
 */
public class ShopPhone implements IShopPhone {
    @Override
    public void shopPhone(String str) {
        System.out.println(str);
    }

}

/**
 * 代理类,代理购买手机的行为。
 */
public class ProxyShopPhone implements IShopPhone {
    IShopPhone mTarget;

    /**
     * 目标
     */
    public ProxyShopPhone(IShopPhone target) {
        this.mTarget = target;
    }

    @Override
    public void shopPhone(String str) {
        System.out.println("-----start----");
        mTarget.shopPhone(str);
        System.out.println("-----end----");
    }
}

/**
 * 用户类,用户进行买手机的操作
 */
public class User {
    public static void main(String[] args) {
        /*在大陆买手机*/
        ShopPhone shopPhone = new ShopPhone();
        shopPhone.shopPhone("在大陆买了一个iPhone X,一共花了8888元");
        /*在香港买手机*/
        ProxyShopPhone proxyShopPhone = new ProxyShopPhone(shopPhone);
        proxyShopPhone.shopPhone("通过代购,在香港买了一个iphone X,一共花了6666元");
    }
}


————————————————————控制台显示的结果—————————————————————

在大陆买了一个iPhone X,一共花了8888元
-----start----
通过代购,在香港买了一个iphone X,一共花了6666元
-----end----

Process finished with exit code 0

——————————————END——————————————


动态代理

与静态代理的区别

动态创建代理类(虚拟机->框架、系统帮助我们来完成创建过程)

动态代理的特点

1.代理对象不需要实现接口。
2.不需要自己实现代理对象,由虚拟机动态生成(内部通过java反射实现)。
3.动态代理也叫做JDK代理或接口代理。

代码实现

将动态代理中,不使用静态代理中的代理对象,通过jdk中自带的代理方法实现动态代理。


/**
 * 操作类---测试类
 */
public class Main {
    public static void main(String[] args) {
        /*在大陆买手机*/
        ShopPhone shopPhone = new ShopPhone();
        shopPhone.shopPhone("在大陆买了一个iPhone X,一共花了8888元");
        /*动态代理---在香港买手机*/
        IShopPhone proxy = (IShopPhone) Proxy.newProxyInstance(shopPhone.getClass().getClassLoader(), shopPhone.getClass().getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                System.out.println("-----动态代理 start----");
                Object returnValue = method.invoke(shopPhone, args);
                System.out.println("-----动态代理 end----");
                return returnValue;
            }
        });
        proxy.shopPhone("通过代购,在香港买了一个iphone X,一共花了6666元");
    }

  • newProxyInstance详解:

ClassLoader loader: 指定当前目标对象使用类加载器,即目标对象的类加载器。获取方法是固定的:[shopPhone.getClass().getClassLoader()]

Class<?>[] interfaces: 目标对象实现的接口的类型,使用泛型方式确认类型。获取方法是固定的:[shopPhone.getClass().getInterfaces()]

InvocationHandler h: 事件处理,执行目标对象的方法时,会触发事件处理器的方法,会把当前执行目标对象的方法作为参数传入。即动态代理类接口的实现,通过JAVA反射实现。

封装动态代理工厂代码实现


/**
 * 动态代理工厂
 */
public class ProxyFactory {
    private Object mTarget;

    /**
     * 维护一个目标对象
     */
    public ProxyFactory(Object target) {
        this.mTarget = target;
    }

    /**
     * 获取动态代理对象
     *
     * @return
     */
    public Object getProxyInstance() {
        return Proxy.newProxyInstance(mTarget.getClass().getClassLoader(), mTarget.getClass().getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                System.out.println("--------start-------");
                Object returnValue = method.invoke(mTarget,args);
                System.out.println("---------end--------");
                return returnValue;
            }
        });
    }
}


——————————————————————使用方法—————————————————————
 proxy.shopPhone("通过代购,在香港买了一个iphone X,一共花了6666元");
        IShopPhone proxy2 = (IShopPhone) new ProxyFactory(shopPhone).getProxyInstance();
        proxy2.shopPhone("通过代购商店,在香港买了一个iphone X,一共花了6666元");

——————————————————————控制台输出结果—————————————————————
在大陆买了一个iPhone X,一共花了8888元
--------start-------
通过代购商店,在香港买了一个iphone X,一共花了6666元
---------end--------

Process finished with exit code 0

cglib动态代理

基本概述

可继承式的动态代理(java仅允许单继承,而JDK中的代理类,自身就会继承Proxy),在android开发中,一般不会使用。
使用该方式实现动态代理,需要倒入cglib.jar,或使用spring框架,即该方式的动态代理,大多在后台服务器中使用,在本片文章中不会过多讲解;

代码实现

/**

/**
 * 通过继承实现的动态代理类
 */
public class CGLibProxy implements MethodInterceptor {
    private Enhancer enhancer = new Enhancer();
    public Object getProxy(Class<?> clazz){
        enhancer.setSuperclass(clazz);
        enhancer.setCallback(this);
        return enhancer.create();
    }
    /**
     * 拦截所有目标类方法的调用
     * 参数:
     * obj目标实例对象
     *method 目标方法的反射对象
     * args方法的参数
     * proxy代理类的实例
     */
    public Object intercept(Object obj, Method method, Object[] args,
                            MethodProxy proxy) throws Throwable {
        //代理类调用父类的方法
        System.out.println("日志开始");
        proxy.invokeSuper(obj, args);
        System.out.println("日志结束");
        return null;
    }
}
——————————————————————使用方法—————————————————————
 /*通过cglib.jar实现可继承的动态代理*/
    IShopPhone proxy3 = (IShopPhone) new CGLibProxy().getProxy(ShopPhone.class);
    proxy3.shopPhone("通过代购,在香港买了一个iphone X,一共花了6666元");
       

使用场景

例如:开发当中->类似框架(代理模式)
    XUtils框架、Retrofit框架、MVP架构设计、插件化架构设计等等...

最后,为自己留一个作业:如何通过java反射自己实现JDK动态代理?

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