黑马程序员技术交流社区

标题: 【上海校区】这是一份全面 & 清晰的动态代理模式学习指南 [打印本页]

作者: wuqiong    时间: 2018-6-7 11:17
标题: 【上海校区】这是一份全面 & 清晰的动态代理模式学习指南
本帖最后由 wuqiong 于 2018-6-7 11:37 编辑

目录


1. 为什么要使用动态代理1.1 背景

代理模式中的静态代理模式存在一些特点:



关于静态代理模式可以看文章:代理模式(Proxy Pattern):静态代理 - 最易懂的设计模式解析

1.2 冲突

在目标对象较多的情况下,若采用静态代理,则会出现 静态代理对象量多、代码量大,从而导致代码复杂的问题

1.3 解决方案

采用 动态代理模式。


2. 动态代理模式介绍2.1 实现原理
2.2 优点
2.3 缺点2.4 应用场景
2.5 与静态代理模式的区别


3. 具体应用接下来,我将用1个具体实例来对 动态代理模式 进行更深一步的介绍3.1 实例概况
3.2 使用步骤3.3 步骤详解

步骤1: 声明 调用处理器类

DynamicProxy.java


<-- 作用 -->

// 1.  生成 动态代理对象

// 2.  指定 代理对象运行目标对象方法时需要完成的 具体任务

// 注:需实现InvocationHandler接口 = 调用处理器 接口

// 所以称为 调用处理器类


public class DynamicProxy implements InvocationHandler {


    // 声明代理对象

    // 作用:绑定关系,即关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke()

    private Object ProxyObject;


    public Object newProxyInstance(Object ProxyObject){

        this.ProxyObject =ProxyObject;

        return Proxy.newProxyInstance(ProxyObject.getClass().getClassLoader(),

                ProxyObject.getClass().getInterfaces(),this);

        // Proxy类 = 动态代理类的主类

        // Proxy.newProxyInstance()作用:根据指定的类装载器、一组接口 & 调用处理器 生成动态代理类实例,并最终返回

        // 参数说明:

        // 参数1:指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器

        // 参数2:指定目标对象的实现接口

        // 即要给目标对象提供一组什么接口。若提供了一组接口给它,那么该代理对象就默认实现了该接口,这样就能调用这组接口中的方法

        // 参数3:指定InvocationHandler对象。即动态代理对象在调用方法时,会关联到哪个InvocationHandler对象


    }


    //  复写InvocationHandler接口的invoke()

    //  动态代理对象调用目标对象的任何方法前,都会调用调用处理器类的invoke()

    @Override

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

            // 参数说明:

            // 参数1:动态代理对象(即哪个动态代理对象调用了method()

            // 参数2:目标对象被调用的方法

            // 参数3:指定被调用方法的参数

            throws Throwable {

                System.out.println("代购出门了");

                Object result = null;

                // 通过Java反射机制调用目标对象方法

                result = method.invoke(ProxyObject, args);

        return result;

    }


}




步骤2: 声明目标对象的抽象接口


Subject.java


public interface Subject {

    // 定义目标对象的接口方法

    // 代购物品

    public  void buybuybuy();


}


步骤3: 声明目标对象类


Buyer1.java



// 小成,真正的想买Mac的对象 = 目标对象 = 被代理的对象

// 实现抽象目标对象的接口

public class Buyer1 implements Subject  {


    @Override

    public void buybuybuy() {

        System.out.println("小成要买Mac");

    }


}


Buyer2.java


// 小何,真正的想买iPhone的对象 = 目标对象 = 被代理的对象

// 实现抽象目标对象的接口

public class Buyer2 implements Subject  {


    @Override

    public void buybuybuy() {

        System.out.println("小何要买iPhone");

    }


}


步骤4: 通过动态代理对象,调用目标对象的方法

MainActivity.java


public class MainActivity extends AppCompatActivity {


    @Override

    protected void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);


        // 1. 创建调用处理器类对象

        DynamicProxy DynamicProxy = new DynamicProxy();


        // 2. 创建目标对象对象

        Buyer1 mBuyer1 = new Buyer1();


        // 3. 创建动态代理类 & 对象:通过调用处理器类对象newProxyInstance()

        // 传入上述目标对象对象

        Subject Buyer1_DynamicProxy = (Subject) DynamicProxy.newProxyInstance(mBuyer1);


        // 4. 通过调用动态代理对象方法从而调用目标对象方法

        // 实际上是调用了invoke(),再通过invoke()里的反射机制调用目标对象的方法

        Buyer1_DynamicProxy.buybuybuy();

        // 以上代购为小成代购Mac


        // 以下是代购为小何代购iPhone

        Buyer2 mBuyer2 = new Buyer2();

        Subject Buyer2_DynamicProxy = (Subject) DynamicProxy.newProxyInstance(mBuyer2);

        Buyer2_DynamicProxy.buybuybuy();

    }

}

3.4 测试结果


3.5 Demo地址

Carson_Ho的Github地址:动态代理DynamicProxy


4. 源码分析

至此,关于代理模式中的动态代理模式的相关知识已经讲解完毕。


5. 总结
  • 我用两张图总结整篇文章的内容





作者: 吴琼老师    时间: 2018-7-5 17:45

作者: 不二晨    时间: 2018-7-17 14:17
棒棒哒




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2