Java代理系列-动态代理热身

妖精的绣舞 提交于 2020-03-02 08:20:32

上一章学习了Java的静态代理,但是我们主要用的是动态代理,在学习动态代理前,先熟悉一下需要的API

InvocationHandler

Proxy

Method

InvocationHandler是使用Java动态代理必须实现(implements)的接口,它只有一个方法invoke

可能使用过java反射的朋友们都见过此方法,它是用来调用最终的方法的。

下面贴出Java JDK文档对此方法的详细解释

invoke


Object invoke(Object proxy, Method method, Object[] args)

              throws Throwable

在代理实例上处理方法调用并返回结果。在与方法关联的代理实例上调用方法时,将在调用处理程序上调用此方法。

参数:

proxy - 在其上调用方法的代理实例

method - 对应于在代理实例上调用的接口方法的 Method 实例。Method 对象的声明类将是在其中声明方法的接口,该接口可以是代理类赖以继承方法的代理接口的超接口。

args - 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 null。基本类型的参数被包装在适当基本包装器类(如 java.lang.Integer 或 java.lang.Boolean)的实例中。

返回:

从代理实例的方法调用返回的值。如果接口方法的声明返回类型是基本类型,则此方法返回的值一定是相应基本包装对象类的实例;否则,它一定是可分配到声明返回类型的类型。如果此方法返回的值为 null 并且接口方法的返回类型是基本类型,则代理实例上的方法调用将抛出 NullPointerException。否则,如果此方法返回的值与上述接口方法的声明返回类型不兼容,则代理实例上的方法调用将抛出 ClassCastException。

抛出:

Throwable - 从代理实例上的方法调用抛出的异常。该异常的类型必须可以分配到在接口方法的 throws 子句中声明的任一异常类型或未经检查的异常类型 java.lang.RuntimeException 或 java.lang.Error。如果此方法抛出经过检查的异常,该异常不可分配到在接口方法的 throws 子句中声明的任一异常类型,代理实例的方法调用将抛出包含此方法曾抛出的异常的 UndeclaredThrowableException。

另请参见:

UndeclaredThrowableException


proxy:要调用方法的接口实现类的实例,比如说上一章的new HelloInterfaceImpl(),它是一个实例对象。

method:你要调用的方法名,必须是HelloInterfaceImpl中有的方法,比如say,你要写个eat,那肯定出错了。

args[]:调用的方法的参数数组

method和args[]参数都不想要自己扩展,只需默认就行。

但是proxy是需要自己注入实现的,其实就是得到它的代理对象实例。

实际调用时,比如这样

//代码不能用,主要看mothod.invoke(proxy, args)
@Override
public Object invoke(Object proxy, Method method, Object[] args)  
            throws Throwable {  
        method.invoke(proxy, args);  
    }

这里还要用到Method代理类,它是用来最后的方法调用,下面是Method接口的invoke方法API

 Object invoke(Object obj, Object... args) 
          对带有指定参数的指定对象调用由此 Method 对象表示的底层方法。

从上可以看出,我们还是只需要得到obj的实例即可,它会返回一个对象。

关于如何进行方法调用呢就写完了,下面看如何得到proxy的实例,这就要使用Proxy接口了,看官方API怎么解释

Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。

它的意思就是说我提供一个静态方法,这个静态方法会产生一个proxy实例。下面是这个方法官方API接口解释

static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) 
          返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。

loader:加载一个要绑定的具体实现类的class

interface:上面的class实现的接口

h:上面的InvocationHandler本身的对象,因为我们要在实现了InvocationHandler 类里得到本身的对象,用this即可,也有其他写法,比如要在测试类里写newProxyInstance方法,那么我们就要把实现类上转型成InvocationHandler对象传进去。这个方法返回的是一个接口,就是实现类implements的那个接口。

了解完这些了呢,下一章该写Java动态代理了。

才疏学浅,如有错误敬请指出,谢谢。





易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!