1. 提前说说
项目中涉及到的代码我都会上传到码云(gitee)或者github上,提供给大家下载参考,文中就以最简单的方式说明执行过程。源码的地址在文末哦!
2. 代理模式
代理模式分为静态代理和动态代理两种方式,静态代理是在开发的时候就写好代理的过程,并且代理类要和目标类实现同一个接口。而动态代理是代理类通过实现InvocationHandler
接口完成,在运行期间动态的构建代理对象,在动态代理的实现过程中还有另一个更为重要的类Proxy
,确的来说,Proxy
负责生成代理对象,而InvocationHandler
是根据生成的代理对象执行增强内容和目标方法或类。
3. 静态代理
3.1 要点
1、代理类需要和目标类需要实现同一个接口
2、在代理类中会内置目标类对象
3.2 代码分析
3.2.1 创建一个接口Transformers(变形金刚)
public interface Transformers {
void trans2man();//变形->人
void trans2car();//变形->车
}
3.2.2 Transformers的实现类TransformersImpl,可以理解为擎天柱
擎天柱实现了变形金刚接口,拥有两个功能分别是变形成人、变形成车。
public class TransformersImpl implements Transformers {
@Override
public void trans2man() {
System.out.println("---->transform to man");
}
@Override
public void trans2car() {
System.out.println("---->transform to car");
}
}
3.2.3 代理类TransformersProxy,和TransformersImpl一样都要实现Transformers接口
public class TransformersProxy implements Transformers {
public Transformers transformers;
public void init(Object transformers) { //初始化
this.transformers = (Transformers) transformers;
}
@Override
public void trans2man() {
System.out.println("---->transform to man before");
transformers.trans2man();
}
@Override
public void trans2car() {
System.out.println("---->transform to car before");
transformers.trans2car();
}
}
3.2.4 运行测试方法,测试代理过程
//静态代理方式
@Test
public void staticProxyTest() {
System.out.println("=========static proxy test start=========");
TransformersProxy proxy = new TransformersProxy();
proxy.init(new TransformersImpl());
proxy.trans2man();
System.out.println("~~~~~~~~~华丽分隔线~~~~~~~~~~");
proxy.trans2car();
System.out.println("=========static proxy test end=========");
}
执行结果:
=========static proxy test start=========
---->transform to man before
---->transform to man
~~~~~~~~~华丽分隔线~~~~~~~~~~
---->transform to car before
---->transform to car
=========static proxy test end=========
先执行代理类中的逻辑,再执行目标方法,这样就完成了代理的过程。如果现在又有一个变形金刚大黄蜂实现了Transformers
类,它的增强内容和擎天柱是相同,这个时候代理类中的增强代码就可复用,只要在初始化代理对象的时候,传入大黄蜂实现类对象即可。
3.3 缺点
随着项目的迭代升级,代理的目标增多,代理的增强内容变多(可能不同的实现类需要增强的内容不同),代理类也会越来越庞大,对整个维护过程也会变得复杂。
4. 动态代理
动态代理用的很是广泛,如面试必问、项目必用的AOP,心中一阵绞痛,想当年因为这问题也在面试中被虐的体无完肤。具体的AOP源码抛开暂时不看,我们所知道的就是它的重要一个实现机制就是动态代理,那就从最基础的了解动态代理的实现。
4.1 代码分析
4.1.1 创建变形金刚接口TransformersDynamic
public interface TransformersDynamic {
void trans2man();//变形->人
void trans2car();//变形->车
}
4.1.2 创建擎天柱TransformersDynamicImpl,实现变形金刚接口
public class TransformersDynamicImpl implements TransformersDynamic {
@Override
public void trans2man() {
System.out.println("---->transform to man");
}
@Override
public void trans2car() {
System.out.println("---->transform to car");
}
}
4.1.3 创建代理类TransformersDynamicProxy,实现InvocationHandler接口
public class TransformersDynamicProxy implements InvocationHandler {
private Object proxyObject;
public TransformersDynamicProxy(Object proxyObject){
this.proxyObject = proxyObject;
}
/**
* 获取代理对象
*/
public Object newProxyInstance() {
return Proxy.newProxyInstance(proxyObject.getClass().getClassLoader(),
proxyObject.getClass().getInterfaces(),
this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("---->dynamic invoke before");
method.invoke(proxyObject, args); //执行目标方法
System.out.println("---->dynamic invoke after");
return null;
}
}
说明:实现InvocationHandler接口后实现invoke接口,这个接口中就定义了这一个接口,源码:
/**
* @author Peter Jones
* @see Proxy
* @since 1.3
*/
public interface InvocationHandler {
/**
* 此处省略100000行注释……
* @see UndeclaredThrowableException
*/
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable;
}
proxy:指我们所代理的真实对象
method:指的是我们所要调用真实对象的某个方法Method对象
args:指调用真实对象某个方法时接收的参数
在newProxyInstance
方法内,通过Proxy
类生成代理对象,第二个参数Interfaces
,是获取当前目标类实现的所有接口。
4.1.4 运行测试方法,测试代理过程
//动态代理方式
@Test
public void dynamicProxyTest() {
System.out.println("=========dynamic proxy test start=========");
TransformersDynamicProxy dynamicProxy = new TransformersDynamicProxy(new TransformersDynamicImpl());
TransformersDynamic proxy = (TransformersDynamic) dynamicProxy.newProxyInstance();
System.out.println("----->" + proxy.getClass());
proxy.trans2man();
System.out.println("~~~~~~~~~华丽分隔线~~~~~~~~~~");
proxy.trans2car();
System.out.println("=========dynamic proxy test end=========");
}
执行结果:
=========dynamic proxy test start=========
----->class com.sun.proxy.$Proxy11
---->dynamic invoke before
---->transform to man
---->dynamic invoke after
~~~~~~~~~华丽分隔线~~~~~~~~~~
---->dynamic invoke before
---->transform to car
---->dynamic invoke after
=========dynamic proxy test end=========
打印的proxy可以看出,这个对象并不是在创建时传入的TransformersDynamicImpl
对象,而是通过Proxy
生成的动态代理对象。
到这里静态代理和动态代理的最基本原理已经说完了。但是这里还是需要说点其他的。
4.2 动态代理的坑
上面动态代理可以看出来,trans2man
和trans2car
都会目标方法,在执行的时候都会执行before
和after
,但是下面这个演示,你将看到不一样的结果。
4.3 代码的改动
1、将dynamicProxyTest方法改成如下内容:
//动态代理方式
@Test
public void dynamicProxyTest() {
System.out.println("=========dynamic proxy test start=========");
TransformersDynamicProxy dynamicProxy = new TransformersDynamicProxy(new TransformersDynamicImpl());
TransformersDynamic proxy = (TransformersDynamic) dynamicProxy.newProxyInstance();
System.out.println("----->" + proxy.getClass());
proxy.trans2man();
System.out.println("=========dynamic proxy test end=========");
}
2、在TransformersDynamicImpl类的trans2man方法中调用trans2car方法:
@Override
public void trans2man() {
System.out.println("---->transform to man");
trans2car();
}
执行结果:
=========dynamic proxy test start=========
----->class com.sun.proxy.$Proxy11
---->dynamic invoke before
---->transform to man
---->transform to car
---->dynamic invoke after
=========dynamic proxy test end=========
从结果分析可以看出来,在输出transform to car前后少了一对before
和after
,也就意味着这个时候trans2car
没有被增强,为什么呢,trans2car
是被增强的啊。
这里需要理解的是,在trans2man
中调用trans2car
方法前面还隐含着一个调用对象,补全就是this.trans2car()
,也就是当前对象调用的trans2car
方法,并不是代理对象调用,那就肯定没有增强逻辑的执行了。代理被绕过,没有生效。在这里有这个问题,那么对于Spring AOP的动态代理有没有问题呢?
5. 源码提供
具体的代码是在com.minuor.staticProxy
包(静态代理示例)、com.minuor.dynamicProxy
包(动态代理示例),测试类在test下com.minuor.MinuorJunitService
。