0 0

动态代理之:接口InvocationHandler 20

前段时间学习JDBC过程中碰到动态代理的相关知识,做测试过程中不能理解接口InvocationHandler 中方法:public Object invoke(Object proxy, Method method, Object[] args) throws Throwable的参数Object proxy做何用,关于动态代理的相关知识就不在此赘述了,现将自己的例子附上,请大家看看参考。

1.接口:
public interface IUser {
public String getName();
}
2.接口对应的类:
public class UserImp implements IUser {
String name;
public UserImp(String name){
this.name = name;
}
public String getName() {
return name;
}

}
3.监听方法所在类,该类实现了InvocationHandler 接口,通过invoke方法利用java的反射机制实现对相关方法的监听。
public class Handler implements InvocationHandler {


public Object targetObj;


public Handler(Object targetObj) {
this.targetObj = targetObj;
}

public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable
{

System.out.println("before the function \""+method.getName()+"\"");
Object ret = method.invoke(targetObj, args);
System.out.println(ret);
System.out.println("after the function \""+method.getName()+"\"");

}

}
4.测试方法:
public class testMain {

public static void main(String[] args) {

IUser realUser = new UserImp("sun");
Handler hand = new Handler(realUser);
IUser proxy = (IUser) Proxy.newProxyInstance(realUser.getClass().getClassLoader(), realUser.getClass().getInterfaces(), hand);
proxy.getName();
}

}
5.输出结果:
before the function "getName"
sun
after the function "getName"


疑问:在步骤3中invoke(Object proxy, Method method, Object[] args)方法的参数proxy的作用是什么,如何使用?
在论坛中看到大家发帖讲述的该参数的作用为:proxy就是你调用的代理
java doc中的说明为:proxy - 在其上调用方法的代理实例

按照上面的叙述,是否可以将步骤3的代码改成如下的形式:
public class Handler implements InvocationHandler {

/*
public Object targetObj;


public Handler(Object targetObj) {
this.targetObj = targetObj;
}
*/

public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable
{

System.out.println("before the function \""+method.getName()+"\"");
Object ret = method.invoke(proxy, args);
System.out.println(ret);
System.out.println("after the function \""+method.getName()+"\"");

}

}

但是改成这样后程序好像进入了死循环一样,如下是输出结果:

at Handler.invoke(Handler.java:27)
at $Proxy0.getName(Unknown Source)
at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:324)
at Handler.invoke(Handler.java:27)
at $Proxy0.getName(Unknown Source)
at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:324)
at Handler.invoke(Handler.java:27)


请各位帮忙答疑!
谢谢!
问题补充
mikewang 写道
没错呀, 自己代理自己, 当然是死循环了。

好像是这么回事,不过这个参数一般如何使用啊。
如果在invoke方法中加入打印它的代码都会出错呢:System.out.println(proxy);

报错如下:
java.lang.StackOverflowError
at Handler.invoke(Handler.java:18)
at $Proxy0.toString(Unknown Source)
at java.lang.String.valueOf(String.java:2131)
at java.io.PrintStream.print(PrintStream.java:462)
at java.io.PrintStream.println(PrintStream.java:599)
at Handler.invoke(Handler.java:18)
at $Proxy0.toString(Unknown Source)
at java.lang.String.valueOf(String.java:2131)
at java.io.PrintStream.print(PrintStream.java:462)
at java.io.PrintStream.println(PrintStream.java:599)
at Handler.invoke(Handler.java:18)

开始以为在执行这段代码时会将真实对象的参数传递到invoke中的proxy参数的:
IUser proxy = (IUser) Proxy.newProxyInstance(realUser.getClass().getClassLoader(), realUser.getClass().getInterfaces(), hand);

这样的话在实现InvocationHandler接口时必须提供传递真实类的方法,也就是上面例子中的:
public Object targetObj;


public Handler(Object targetObj) {
this.targetObj = targetObj;
}

这个和普通的实现接口不一样哈!

2009年4月13日 21:20

4个答案 按时间排序 按投票排序

0 0

public Handler(Object targetObj) {
this.targetObj = targetObj;
}

这个写法无所谓,你也可以用setter getter

2009年4月16日 11:04
0 0

System.out.println(proxy);
很明显就是调用代理的toString,而已这个又是循环调用自己方法,导致异常

2009年4月13日 21:53
0 0

proxy 是你在创建代理的时候有jvm生产的字节码, 文档上已经说的很清楚了, 保证这个proxy 可以转化成你的interface , 但没说这个proxy 就一定是一个可以执行的class

我的理解是 这里的 proxy 类似一个智能指针, 他的作用只不过 可以让jvm从这个指针中的到真正可以执行的代码的位置(就是的hander), 和 函数名称, 参数等信息而已

这点可以从  Proxy.newProxyInstance 的源代码中可以得到证实, 这个函数主要就是调用 getProxyClass 这个函数来创建代理类(也就是你的proxy)

分析这个 getProxyClass 函数, 他做的工作就是
1 整理你需要代理的接口
2 得到每个接口的方法, 参数等信息
3 把每个接口的方法,缓存一下(放到一个map里面, 估计是等到调用的时候, 方便找到)

4 最重要的, 生成proxy  注意, 这里的proxy 核心的部分 就是 刚才说的那个缓存 方法的map

5 最后返回,你要的接口。(这里也不需要判断了, 反正,最后也是你自己在hander里面处理, 只要有方法的名称就可以了)

proxy的执行顺序就是 根据接口找到 函数, 然后在map中找到函数真正的可以执行的代码, 然后在调用传到hander中去执行调用(这步由你完成),实际上就是 5 -> 4 -> 3 -> 2 所以可以看出, proxy 就是我刚才说的, 类似一个智能指针而已。

2009年4月13日 21:20
0 0

没错呀, 自己代理自己, 当然是死循环了。

2009年4月13日 21:20

相关推荐

    Java动态代理实现 Proxy InvocationHandler

    总结来说,Java动态代理通过`Proxy`和`InvocationHandler`提供了灵活的代码扩展机制,可以在运行时为对象创建代理,实现在不修改原有代码的基础上添加额外功能。这在处理需要拦截和增强的对象时非常有用,例如日志...

    动态代理例子

    Proxy类用于创建一个实现了特定接口的代理对象,而InvocationHandler接口定义了一个方法,这个方法会在代理对象调用目标方法时被触发。 1. **Proxy类**: - Proxy类提供了一种机制,可以在运行时创建一个实现了...

    java动态代理例子

    JDK动态代理基于Java接口实现,适用于目标对象实现了至少一个接口的情况。它通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口来工作。Proxy类用于创建代理对象,而InvocationHandler...

    JDK动态代理_JDK动态代理

    3. **创建动态代理对象**:使用`Proxy.newProxyInstance`方法创建动态代理对象,传入ClassLoader、接口数组和上一步骤创建的InvocationHandler实例。 4. **调用代理对象的方法**:当调用代理对象的方法时,会自动...

    动态代理由浅入深

    在动态代理中,InvocationHandler 接口是代理对象的处理器,它负责处理代理对象的方法调用。InvocationHandler 接口中有一个 invoke 方法,该方法是代理对象的核心方法,它负责处理代理对象的方法调用。 在动态代理...

    Spring框架中JDK动态代理和cglib动态代理

    JDK 动态代理是 Java 自带的动态代理机制,它只能代理接口,而不能代理类。这是因为 JDK 动态代理是基于接口的代理,它需要一个接口来生成代理类。如果我们想使用 JDK 动态代理,必须提供一个接口,并且将其实现类...

    java代理机制 JDK动态代理和cglib代理 详解

    - `MethodInterceptor`接口类似于JDK动态代理中的`InvocationHandler`,但它的拦截逻辑更为灵活,可以在`intercept()`方法中控制目标方法的调用。 3. **FastClass和MethodProxy** - CGLIB通过`FastClass`和`...

    jdk动态代理技术详解

    InvocationHandler 接口是 JDK 中和动态代理直接相关的主要接口之一。该接口相当于 Proxy 类的 CallBack Interface,定义了一个方法 Object:invoke(Object obj, Method method, Object[] args)。第一个参数 proxy ...

    spring+动态代理

    动态代理的核心概念是JDK的`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口。Proxy类用于创建一个代理对象,而InvocationHandler接口定义了处理代理对象方法调用的逻辑。Spring框架提供了...

    Jdk动态代理和cglib动态代理原理

    2. **Proxy类**:用于创建动态代理对象,通过`newProxyInstance()`静态方法,传入类加载器、接口数组和InvocationHandler实例来生成代理对象。 3. **实现过程**:首先,你需要定义一个接口,然后创建实现该接口的...

    代理设计模式:静态代理和动态代理的理解、实现与区别(优缺点)与SpringAOP的3种配置方式案例工程代码

    动态代理通常使用Java的`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口实现。在运行时,我们可以动态地创建代理对象,而无需预先编写代理类。 优点: 1. 不需要为每个目标类创建单独的...

    关于jdk动态代理的源码剖析

    `InvocationHandler`是动态代理的核心接口,其主要作用是处理代理对象的方法调用。当通过代理对象调用一个方法时,该方法的调用会被委托给`InvocationHandler`实例的`invoke`方法处理。 ```java public interface ...

    浅谈JDK动态代理与CGLIB代理去区别

    首先,JDK动态代理主要依赖于java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口。Proxy类用于创建一个代理对象,而InvocationHandler接口则定义了处理代理对象调用方法的逻辑。当调用代理对象的...

    spring动态代理

    动态代理主要涉及两个核心接口:`java.lang.reflect.InvocationHandler` 和 `java.lang.Proxy`。 1. **InvocationHandler接口**: `InvocationHandler` 是 Java 核心库中的接口,用于处理代理对象上的方法调用。当...

    jdk与cglib动态代理与底层实现

    - **JDK代理**:JDK动态代理使用反射和InvocationHandler接口生成一个实现了目标接口的新类。在运行时,这个新类的实例作为代理对象,它的方法调用都会转发到InvocationHandler的`invoke()`方法。 - **CGLIB代理**...

    动态代理与RMI远程调用

    2. 创建InvocationHandler:实现`InvocationHandler`接口,重写`invoke`方法,这里可以添加自定义逻辑。 3. 创建代理对象:使用`Proxy.newProxyInstance()`方法,传入类加载器、接口列表和自定义的`...

    动态代理设计模式 日志和源码

    你可以查看定义的接口、代理类、InvocationHandler的实现,以及Spring配置文件,了解它们是如何协作完成动态代理的。 总之,动态代理设计模式是软件开发中一种强大的工具,它让我们能够在不改变原有代码的情况下...

    InvocationHandler, Proxy机制

    在Java编程语言中,`InvocationHandler`和`Proxy`机制是动态代理的重要组成部分,它们提供了在运行时创建接口实现的能力,使得我们可以在不修改原有代码的情况下扩展或增强对象的行为。这些概念对于理解和实现面向切...

    JDK动态代理简单示例

    JDK动态代理是Java编程中一个非常重要的特性,它允许我们在运行时创建具有特定接口的代理类实例。这种技术在很多场景下都非常有用,比如在AOP(面向切面编程)中实现方法拦截、日志记录、事务管理等。下面我们将深入...

    模拟JDK动态代理内部实现

    Java中的动态代理通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口实现。`Proxy`类用于创建一个实现了特定接口的代理对象,而`InvocationHandler`接口定义了处理代理对象方法调用的逻辑...

Global site tag (gtag.js) - Google Analytics