您的位置:首页 > 其它

动态代理解析

2016-07-14 17:43 183 查看

retrofit中动态代理的解析

问题

学习动态代理对我们有什么用?

动态代理是什么?

有什么作用?适用场景在哪里?

学习作用:了解retrofit源码。

动态代理是什么?运行时动态生成代理类。既然是运行时动态生成,那么说明其灵活性大,作用域广。

动态代理生成的方式有多种,这回探讨的是JDK自带的方法。也就是实现InvocationHandler接口,重写invoke()方法,再通过Proxy.newProxyInstance()方法调用。

如retrofit源码:

Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] { service },new InvocationHandler() {
private final Platform platform = Platform.get();

@Override public Object invoke(Object proxy, Method method, Object... args)
throws Throwable {
......//省略若干代码
ServiceMethod serviceMethod = loadServiceMethod(method);
OkHttpCall okHttpCall = new OkHttpCall<>(serviceMethod, args);
return serviceMethod.callAdapter.adapt(okHttpCall);
}
});
}


Interface a = Proxy.newProxyInstance(ClassLoader,Interface.class,new InvocationHandler()),这段其实生成了a的子类,调a接口的方法时会调用InvocationHandler类的invoke(..method..)方法,也就是重写的invoke方法。

newProxyInstance()方法完成了三个步骤:1 获取Proxy类getProxyClass(classloader,interface) 。2 通过反射获得构造函数对象 getConstructor(InvocationHandler.class) 3 实例化构造函数 newInstance(InvocationHandler)。一般来说通过反射实例化一个类,也莫过于此,通用的三步走,而动态代理重点在于第一步,获取Proxy类,里面干货满满啊。

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,
InvocationHandler invocationHandler)
throws IllegalArgumentException {

......//省略若干代码
try {
return getProxyClass(loader, interfaces)
.getConstructor(InvocationHandler.class)
.newInstance(invocationHandler);
} catch (NoSuchMethodException e) {
cause = e;
} catch (IllegalAccessException e) {
cause = e;
} catch (InstantiationException e) {
cause = e;
} catch (InvocationTargetException e) {
cause = e;
}
...
}


为什么会说动态代理是运行时动态生成的?

原因是代理类的字节码在运行时才生成并载入java虚拟机中。在jvm运行时类加载器ClassLoader类进行加载的。

动态代理最神奇的地方可能不在于它对被委托类似真似假的代理,而在于它能像魔术一样,凭空生成一个之前并不存在的类。它究竟是如何做到的?

关于动态生成代理类的关键,在于getProxyClass方法里面,但当你进去看了之后,会发现一个神秘的闭源方法ProxyGenerator.generateProxyClass(),属于未公开的sun.misc包。然而以上是我从文章上看到的,我自己追查了源码,只找到了generateProxy(…) Native方法(我是jdk1.8估计源码被修改了),我猜大致意思就是全转为字节码,运行时(runtime)建proxy。其实ClassLoader也该研究研究了。

问题来了,为啥代理类只接受interface的代理?因为所有凭空生成的Proxy的儿子$ProxyN只能有一个父亲,并且必须实现传人的interface。如果class代理的话,那就多继承乱套了。

其实我最关心的还是动态代理我可以用于什么地方,我怎么样才能用它提高能力(逼格)?

据说动态代理的好处是可以减少手写代码。

也不要只关心了动态,忘掉了代理。代理也是很有前途的娃娃。 它可以进行远程代理,虚拟代理,智能引用等等这些名词。其实就是可以当中介,那什么时候应该用呢?比如想要懒加载的时候,先加载中介这货,当真正用到时才加载真正的委托类。或者对某个方法不爽,想对那个方法加点料但又不能改变那个方法时。

最后我发现,研究了那么多,并没有什么毛用,还是不知道该用在哪儿,动态代理这玩意儿能用的地方我差不多都能用其它方式实现,看来我能力还不够啊 23333333 我只能说西湖的水我的泪了。

参考文章:

Java 动态代理机制分析及扩展,第 1 部分

Java 理论与实践: 用动态代理进行修饰

公共技术点之 Java 动态代理

Retrofit2源码分析
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  动态代理 retrofit