动态代理
Posted beanbag
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了动态代理相关的知识,希望对你有一定的参考价值。
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
在调度真实对象之后服务
以上是关于动态代理的主要内容,如果未能解决你的问题,请参考以下文章