码迷,mamicode.com
首页 > 其他好文 > 详细

四、动态代理

时间:2018-11-11 14:17:00      阅读:164      评论:0      收藏:0      [点我收藏+]

标签:getc   调度   out   商务   throw   软件工程   关系   tin   null   

1、理解代理模式

假设读者您目前就职于一家软件公司,担任软件工程师角色。客户带着需求来到你们公司,显然客户不会直接和你进行交流,而是去找商务,此时客户会认为商务就代表公司。此时商务就可以看成代理对象,而您,伟大的软件工程师,就可以看成一个真实的对象。

商务这个角色存在的意义就在于商务可以进行谈判。比如客户和商务谈判软件的价格、交付、开发进度等。这些事情都不要我们软件工程师参与。因此,代理的作用就是:在访问真实对象之前或者之后,加入一定的逻辑,根据其他规则来控制是否使用真实对象。

所以,商务和软件工程师之间的关系就是代理和被代理的关系。客户是经过商务去访问软件工程师的,此时客户就是程序中的调用者,商务就是代理对象,软件工程师就是真实对象。我们需要在调用者调用对象之前产生一个代理对象,而这个代理对象需要和真实对象产生代理关系,所以代理必须分为两个步骤:

  • 代理对象和真实对象产生代理关系。
  • 实现代理对象的代理逻辑方法

2、Java中的代理技术介绍

在Java中有很多的代理技术,比如JDK、CGLIB、Javassist、ASM,其中最常用的动态代理技术有两种,一:JDK的动态代理技术,这种代理技术是基于接口的。二:CGLIB代理技术,这种代理技术不需要提供接口,只需要一个非抽象类就可以实现动态代理。

3、JDK的动态代理

定义一个接口

public interface HelloWorld {
    public void sayHelloWorld();
}

接口的实现类

public class HelloWorldImpl implements HelloWorld {
    @Override
    public void sayHelloWorld() {
        System.out.println("hello world");
    }
}

动态代理绑定和代理逻辑的实现

按照之前的分析,需要先建立代理对象和真实对象之间的关系,然后实现代理逻辑。

在JDK的动态代理中,要想实现代理逻辑,必须必须去实现java.lang.reflect.InvocationHandler接口 它里面定义了一个invoke方法,并提供接口数组用于下挂代理对象。代码如下:

public class JDKProxyExample implements InvocationHandler {

    //真实对象
    private Object target = null;

    /**
     * 建立代理对象和真实对象之间的关系,并返回代理对象
     * @param target 真实对象
     * @return 代理对象
     */
    public Object bind(Object target){
        this.target = target;
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
    }

    /**
     *
     * @param proxy 代理对象
     * @param method 当前调度的方法
     * @param args 方法参数列表
     * @return 代理结果返回
     * @throws Throwable 异常
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("进入代理逻辑方法");
        System.out.println("在调度真实对象之前服务");
        Object invoke = method.invoke(target, args);//相当于调度sayHelloworld方法
        System.out.println("在调度真实对象之后服务");
        return invoke;
    }

    @Test
    public void testJdkProxy(){
        JDKProxyExample jdkProxyExample = new JDKProxyExample();
        HelloWorld proxy = (HelloWorld) jdkProxyExample.bind(new HelloWorldImpl());
        proxy.sayHelloWorld();
    }

}

上述代码中,首先通过target来保存真实的对象,然后通过代码

Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);

来生成代理对象,并返回。方法中包含三个参数

  • 第一个是类加载器,我们采用target本身的类加载器。
  • 第二个是把生成的代理对象下挂在那些接口之下,这样写我们就是放在target实现的接口之下。
  • 第三个就是定义实现方法逻辑的代理类,this代表当前对象,它必须实现InvocationHandler接口的invoke方法,它就是代理逻辑方法的现实方法。

nvoke方法实现了代理逻辑,invoke方法的三个参数的含义如下所示:

  • proxy,代理对象,就是bind方法生成的对象
  • method,当前调度的方法。
  • args,调度方法的参数。 当我们使用代理对象调度方法以后,它就会进入到invoke方法里面。
Object invoke = method.invoke(target, args);//相当于调度sayHelloworld方法

这行代码相当于调度真实对象的方法,只是通过反射实现而已。 类比上面的例子,proxy相当于商务对象,target就是软件工程师对象,bind方法就是建立商务对象和软件工程师对象之间的关系。而invoke就是商务逻辑,它将控制对于软件工程师的访问逻辑。

结果

进入代理逻辑方法
在调度真实对象之前服务
hello world
在调度真实对象之后服务

 

四、动态代理

标签:getc   调度   out   商务   throw   软件工程   关系   tin   null   

原文地址:https://www.cnblogs.com/beanbag/p/9941958.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!