InvocationHandler的invoke方法如何被调用?

<wbr></wbr>

在客户类(即以下代码的Client类),语句Manager managerProxy =

<wbr> (Manager) Proxy.newProxyInstance(managerImpl.getClass().getClassLoader(), managerImpl.getClass().getInterfaces(), securityHandler);</wbr>

<wbr></wbr>

<wbr> 可以看到,Proxy(库类)中的newProxyInstance方法被调用,该方法返回一个被代理对象的实例,然后向上转型为其对应的接口。</wbr>

<wbr></wbr>

<wbr> 问题是该方法在返回之前已经做了什么?我们看看其参数的第三个:securityHandler,它就是InvocationHandler接口实现类的一个实例:securityHandler。因为InvocationHandler接口中原本有个invoke的方法,所以其实现类当然需要实现这个方法,即其实现类(在此是BusinessHandler类)中有invoke方法,而如果invoke方法要被调用,只能通过BusinessHandler类对象来调用。</wbr>

<wbr></wbr>

<wbr> 而在下面的源代码类$Proxy0 中,有三个地方已经用BusinessHandler类对象来调用invoke方法,他们分别在这些方法的代码中:public final boolean equals(Object obj) ;public final int hashCode() ;public final String toString()。也许你会有疑问,这个方法也没有看到在哪里被</wbr>

$Proxy0 对象调用过,怎么能执行invoke方法呢?但是请看$Proxy0中的static代码块,这个模块是特殊的,因为当newProxyInstance创建$Proxy0 时,它就被初始化。而这个static模块中的getMethod方法加载了这个三个方法,因而它们里面的代码(h.invoke())被执行。

<wbr></wbr>

<wbr> 另外我们还可以看到,invoke方法中的第一个参数代表什么。在源代码中我们看到return ((Boolean) super.h.invoke(this, m1, new Object[] { obj })) .booleanValue(); 那么this就是$Proxy0 对象。这里需要注意的是,它并非是被代理对象(ManagerImpl)。其第二个参数传递的是$Proxy0 类中数据域,其类型是Method,是被封装过的被代理对象(ManagerImpl)的方法。</wbr>

<wbr> 小结:invoke方法的调用过程,就是先新建其类对象(实例),然后把它传入newProxyInstance方法中,在里面解析并用它来调用invoke方法。</wbr>

<wbr></wbr>

<wbr></wbr>

一.相关类及其方法:

java.lang.reflect.Proxy类的newProxyInstance(),是用于创建动态代理类和实例的静态方法.返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。

java.lang.reflect.InvocationHandler接口中的invoke(),在代理实例上处理方法调用并返回结果。当与方法关联的代理实例上调用方法时,将在调用处理程序上调用此方法。

二.探讨的源代码:

被代理对象的接口及实现类:

public interface Manager {
public void modify();
}

<wbr></wbr>

public class ManagerImpl implements Manager {
<wbr> public void modify() {<br><wbr><wbr><wbr><wbr>System.out.println("*******modify()方法被调用");<br><wbr> }<br> }</wbr></wbr></wbr></wbr></wbr></wbr>

<wbr></wbr>

动态代理类:

<wbr></wbr>

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

<wbr></wbr>

public class BusinessHandler implements InvocationHandler {

private Object object = null;

public BusinessHandler(Object object) {
<wbr><wbr> this.object = object;<br> }</wbr></wbr>

@Override
public Object invoke(Object proxy, Method method, Object[] args)throws Throwable {
<wbr><wbr><wbr> System.out.println("do something before method");<br><wbr><wbr><wbr><wbr>Object ret = method.invoke(this.object, args);<br><wbr><wbr><wbr> System.out.println("do something after method");<br><wbr><wbr><wbr> return ret;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>

<wbr><wbr><wbr>}<br> }</wbr></wbr></wbr>



客户端类:
import java.lang.reflect.Proxy;


public class Client {

public static void main(String[] args) {
<wbr><wbr> // 元对象(被代理对象)<br><wbr><wbr> ManagerImpl managerImpl = new ManagerImpl();</wbr></wbr></wbr></wbr>

<wbr><wbr> // 业务代理类<br><wbr><wbr> BusinessHandler securityHandler = new BusinessHandler(managerImpl);</wbr></wbr></wbr></wbr>

<wbr><wbr> // 获得代理类($Proxy0 extends Proxy implements Manager)的实例.<br><wbr><wbr> Manager managerProxy =</wbr></wbr></wbr></wbr>

<wbr> (Manager) Proxy.newProxyInstance(managerImpl.getClass().getClassLoader(), managerImpl.getClass()<br><wbr><wbr><wbr><wbr> .getInterfaces(), securityHandler);</wbr></wbr></wbr></wbr></wbr>

<wbr><wbr> managerProxy.modify();<br><wbr> }<br> }</wbr></wbr></wbr>

<wbr></wbr>

三.执行结果:
do something before method
*******modify()方法被调用
do something after method

<wbr></wbr>

四.机制分析:

Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)做了以下几件事.
(1)根据参数loader和interfaces调用方法 getProxyClass(loader, interfaces)创建代理类$Proxy.$Proxy0类 实现了interfaces的接口,并继承了Proxy类.
(2)实例化$Proxy0并在构造方法中把BusinessHandler传过去,接着$Proxy0调用父类Proxy的构造器,为h赋值,如下:
class Proxy{
<wbr><wbr> InvocationHandler h=null;<br><wbr><wbr> protected Proxy(InvocationHandler h) {<br><wbr><wbr><wbr> this.h = h;<br><wbr><wbr> }<br><wbr><wbr> ...<br> }<br><br><br><br> 下面是本例的$Proxy0类的源码(好不容易才把它提出来):</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableExcep<wbr>tion;</wbr>

<wbr></wbr>

public final class $Proxy0 extends Proxy implements Manager {

private static Method m1;
private static Method m0;
private static Method m3;
private static Method m2;

static {
<wbr><wbr> try {<br><wbr><wbr><wbr> m1 = Class.forName("java.lang.Object").getMethod("equals",<br><wbr><wbr><wbr><wbr><wbr> new Class[] { Class.forName("java.lang.Object") });</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>


<wbr><wbr><wbr> m0 = Class.forName("java.lang.Object").getMethod("hashCode",<br><wbr><wbr><wbr><wbr><wbr> new Class[0]);</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>


<wbr><wbr><wbr> m3 = Class.forName("com.ml.test.Manager").getMethod("modify",<br><wbr><wbr><wbr><wbr><wbr> new Class[0]);</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>


<wbr><wbr><wbr> m2 = Class.forName("java.lang.Object").getMethod("toString",<br><wbr><wbr><wbr><wbr><wbr> new Class[0]);</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>


<wbr><wbr> } catch (NoSuchMethodException nosuchmethodexception) {<br><wbr><wbr><wbr> throw new NoSuchMethodError(nosuchmethodexception.getMessage());<br><wbr><wbr> } catch (ClassNotFoundException classnotfoundexception) {<br><wbr><wbr><wbr> throw new NoClassDefFoundError(classnotfoundexception.getMessage());<br><wbr><wbr> }<br> } //static</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>


public $Proxy0(InvocationHandler invocationhandler) {
<wbr><wbr> super(invocationhandler);<br> }</wbr></wbr>

<wbr></wbr>

@Override
public final boolean equals(Object obj) {
<wbr><wbr> try {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr> return ((Boolean) super.h.invoke(this, m1, new Object[] { obj })) .booleanValue();<br><wbr><wbr><wbr><wbr><wbr><wbr>} catch (Throwable throwable) {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>throw new UndeclaredThrowableExcep<wbr>tion(throwable);<br><wbr><wbr><wbr><wbr><wbr><wbr> }<br> }</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>

<wbr></wbr>

@Override
public final int hashCode() {
<wbr><wbr> try {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr> return ((Integer) super.h.invoke(this, m0, null)).intValue();<br><wbr><wbr><wbr><wbr><wbr><wbr> } catch (Throwable throwable) {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>throw new UndeclaredThrowableExcep<wbr>tion(throwable);<br><wbr><wbr><wbr><wbr><wbr><wbr>}<br> }</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>

<wbr></wbr>

public final void modify() {
<wbr><wbr> try {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr> super.h.invoke(this, m3, null);<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr> return;<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr> } catch (Error e) {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr> } catch (Throwable throwable) {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr> throw new UndeclaredThrowableExcep<wbr>tion(throwable);<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr>}<br> }</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>

<wbr></wbr>

@Override
public final String toString() {
<wbr><wbr> try {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr> return (String) super.h.invoke(this, m2, null);<br><wbr><wbr><wbr><wbr><wbr><wbr> } catch (Throwable throwable) {<br><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr> throw new UndeclaredThrowableExcep<wbr>tion(throwable);<br><wbr><wbr><wbr><wbr><wbr><wbr> }<br><wbr>}<br> }</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>

<wbr><wbr> 接着把得到的$Proxy0实例强制转换成Manager.当执行managerProxy.modify()方法时(managerProxy是Manager的一个已经向上转型的实例),就调用了$Proxy0类中的modify()方法.</wbr></wbr>


<wbr><wbr> 在modify方法中,调用父类Proxy中的h的invoke()方法.即InvocationHandler.invoke();</wbr></wbr>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值