`
jadylen
  • 浏览: 3986 次
  • 性别: Icon_minigender_1
  • 来自: 无锡
最近访客 更多访客>>
文章分类
社区版块
存档分类
最新评论

java 动态代理原理(Proxy,InvocationHandler),含$Proxy0源码

阅读更多
今天没事,把以前的知识回顾一下,网上找了找,发现有一篇比较适合。
概述:其实JDK的动态代理,实际上就是“反射”与“执行时动态生成字节码”二者的结合体;就spring的AOP而言也是用的JDK的动态代理(当然也有cglib方式)。
以下来自网络上的博客:

一.相关类及其方法:
java.lang.reflect.Proxy,
Proxy 提供用于创建动态代理类和实例的静态方法.
newProxyInstance()
返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序
(详见api文档)

java.lang.reflect.InvocationHandler,
InvocationHandler 是代理实例的调用处理程序 实现的接口。
invoke()
在代理实例上处理方法调用并返回结果。在与方法关联的代理实例上调用方法时,将在调用处理程序上调用此方法。
(详见api文档)
二.源代码:
被代理对象的接口及实现类:
package com.ml.test; 
 
public interface Manager { 
public void modify(); 
} 
 
package com.ml.test; 
 
public class ManagerImpl implements Manager { 
 
@Override 
public void modify() { 
   System.out.println("*******modify()方法被调用"); 
} 
} 
业务代理类:
package com.ml.test; 
 
import java.lang.reflect.InvocationHandler; 
import java.lang.reflect.Method; 
 
public class BusinessHandler implements InvocationHandler { 
 
private Object object = null; 
 
public BusinessHandler(Object object) { 
   this.object = object; 
} 
 
@Override 
public Object invoke(Object proxy, Method method, Object[] args) 
    throws Throwable { 
   System.out.println("do something before method"); 
   Object ret = method.invoke(this.object, args); 
   System.out.println("do something after method"); 
   return ret; 
 
} 
} 

客户端类:
 
package com.ml.test; 
import java.lang.reflect.Proxy; 
public class Client { 
 
public static void main(String[] args) { 
   // 元对象(被代理对象) 
   ManagerImpl managerImpl = new ManagerImpl(); 
 
   // 业务代理类 
   BusinessHandler securityHandler = new BusinessHandler(managerImpl); 
 
   // 获得代理类($Proxy0 extends Proxy implements Manager)的实例. 
   Manager managerProxy = (Manager) Proxy.newProxyInstance(managerImpl 
     .getClass().getClassLoader(), managerImpl.getClass() 
     .getInterfaces(), securityHandler); 
 
   managerProxy.modify(); 
} 
} 

三.执行结果:
do something before method
*******modify()方法被调用
do something after method
四.机制分析:
Proxy.(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)做了以下几件事.
(1)根据参数loader和interfaces调用方法 getProxyClass(loader, interfaces)创建代理类$Proxy.
$Proxy0类实现了interfaces的接口,并继承了Proxy类.
(2)实例化$Proxy0并在构造方法中把BusinessHandler传过去,接着$Proxy0调用父类Proxy的构造器,为h赋值,如下:
class Proxy{
   InvocationHandler h=null;
   protected Proxy(InvocationHandler h) {
    this.h = h;
   }
   ...
}



下面是本例的$Proxy0类的源码(好不容易才把它提出来):
 
import java.lang.reflect.InvocationHandler; 
import java.lang.reflect.Method; 
import java.lang.reflect.Proxy; 
import java.lang.reflect.UndeclaredThrowableException; 
 
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 { 
   try { 
    m1 = Class.forName("java.lang.Object").getMethod("equals", 
      new Class[] { Class.forName("java.lang.Object") }); 
    m0 = Class.forName("java.lang.Object").getMethod("hashCode", 
      new Class[0]); 
    m3 = Class.forName("com.ml.test.Manager").getMethod("modify", 
      new Class[0]); 
    m2 = Class.forName("java.lang.Object").getMethod("toString", 
      new Class[0]); 
   } catch (NoSuchMethodException nosuchmethodexception) { 
    throw new NoSuchMethodError(nosuchmethodexception.getMessage()); 
   } catch (ClassNotFoundException classnotfoundexception) { 
    throw new NoClassDefFoundError(classnotfoundexception.getMessage()); 
   } 
} 
 
public $Proxy0(InvocationHandler invocationhandler) { 
   super(invocationhandler); 
} 
 
@Override 
public final boolean equals(Object obj) { 
   try { 
    return ((Boolean) super.h.invoke(this, m1, new Object[] { obj })) 
      .booleanValue(); 
   } catch (Throwable throwable) { 
    throw new UndeclaredThrowableException(throwable); 
   } 
} 
 
@Override 
public final int hashCode() { 
   try { 
    return ((Integer) super.h.invoke(this, m0, null)).intValue(); 
   } catch (Throwable throwable) { 
    throw new UndeclaredThrowableException(throwable); 
   } 
} 
 
public final void modify() { 
   try { 
    super.h.invoke(this, m3, null); 
    return; 
   } catch (Error e) { 
   } catch (Throwable throwable) { 
    throw new UndeclaredThrowableException(throwable); 
   } 
} 
 
@Override 
public final String toString() { 
   try { 
    return (String) super.h.invoke(this, m2, null); 
   } catch (Throwable throwable) { 
    throw new UndeclaredThrowableException(throwable); 
   } 
} 
} 
 

接着把得到的$Proxy0实例强制转换成Manager.
当执行managerProxy.modify()方法时,就调用了$Proxy0类中的modify()方法.
在modify方法中,调用父类Proxy中的h的invoke()方法.
即InvocationHandler.invoke();

以上一段是原作者的原话,在此本人详细解说一下:
             
Manager managerProxy = (Manager) Proxy.newProxyInstance(managerImpl 
     .getClass().getClassLoader(), managerImpl.getClass() 
     .getInterfaces(), securityHandler); 
 

此句中Proxy.newProxyInstance(..)方法执行时生成了$Proxy0的内存字节码文件并return出来赋给了
managerProxy,强制转化成了Manager接口,同时$Proxy0也实现了Manager接口中的所有方法,所以在
managerProxy.modify(); 时就是调用了$Proxy0中的一下代码段:
    
public final void modify() { 
   try { 
    super.h.invoke(this, m3, null); //该段则执行了InvocationHandler.invoke();  super.h既是InvocationHandler
    return; 
   } catch (Error e) { 
   } catch (Throwable throwable) { 
    throw new UndeclaredThrowableException(throwable); 
   } 
} 
 

这样动态代理机制就实现了。
所以JAVA的动态代理的关键就在Proxy.newProxyInstance(..)方法执行时生成了$Proxy0的内存字节码以及JDK的反射机制!
分享到:
评论

相关推荐

    Java 动态代理Proxy应用和底层源码分析.pdf

    通过以上分析,我们可以了解到Java动态代理是如何通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口实现的,以及它的基本应用场景和技术原理。这对于理解和使用AOP、RPC等高级特性有着...

    java Proxy 动态代理

    在Java中,动态代理主要通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口来实现。Proxy类是用于创建一个代理对象,而InvocationHandler接口则定义了代理对象调用方法时的行为。 1. **...

    java InvocationHandler

    Java中的`InvocationHandler`接口是Java动态代理机制的核心组成部分,主要用在`java.lang.reflect`包下。这个接口提供了一种方法,使得我们可以在运行时创建具有特定行为的代理对象。这种行为通常体现在调用代理对象...

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

    ### 关于JDK动态代理的源码剖析 #### 一、引言 在Java开发过程中,动态代理技术是一项非常实用的技术,它可以帮助我们实现在不修改原有代码的基础上为方法增加额外的功能,比如日志记录、权限校验等。本文将深入...

    proxy.rar java三种代理模式源码

    总结来说,"proxy.rar"压缩包提供了Java编程中三种代理模式的源代码实例,通过学习这些代码,开发者可以深入理解代理模式的原理和实现,同时掌握泛型在实际项目中的应用。对于提高代码的可扩展性和可维护性,这些都...

    Java动态代理[动态类Proxy的使用]

    本文将深入探讨Java中的动态代理,特别是基于`java.lang.reflect.Proxy`类的使用。 首先,`java.lang.reflect.Proxy`是Java标准库中用于生成动态代理类的关键类。它允许我们创建一个新的接口实现类,而无需手动编写...

    InvocationHandler, Proxy机制

    通过阅读《JDK的动态代理深入解析(Proxy,InvocationHandler)》和《深入理解Java Proxy机制》文档,你将能够更深入地了解这两个核心概念的内部工作原理,以及如何在实际项目中灵活应用它们。 总之,`...

    Java动态代理Proxy和cglib

    Java的动态代理主要通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口实现。这个主题将深入探讨这两个关键组件以及与之相关的CGLIB库。 ### 1. Java动态代理 #### 1.1 Proxy类 `Proxy`...

    java动态代理机制

    在Java中,动态代理主要通过两个类来实现:`java.lang.reflect.Proxy` 和 `java.lang.reflect.InvocationHandler`。 1. **Proxy类**: `Proxy` 类是Java提供的一个内部类,它主要用于创建一个实现了指定一组接口的...

    Java动态代理(Spring Aop原理)

    Java动态代理是Java提供的一种在运行时创建代理对象的技术,它是通过实现InvocationHandler接口和使用Proxy类来实现的。在Spring框架中,AOP(面向切面编程)就是基于Java动态代理来完成的,用于实现横切关注点,如...

    java代理方法假设和验证的Proxy源码分析.docx

    本文将深入探讨Java代理方法的假设、验证以及Proxy类的源码分析。 首先,让我们理解Java代理的基本概念。在Java中,代理主要通过两种方式实现:接口代理(基于Java动态代理JDK Proxy)和类代理(基于CGLIB等库)。...

    java 静态代理和动态代理学习实例源码

    Java提供了`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口来实现动态代理。 1. **定义接口**:同样,我们首先定义一个接口。 ```java public interface Service { void execute(); } ``...

    java动态代理资源源码

    Java动态代理是Java编程中一个重要的特性,它允许我们在运行时创建代理对象,这些代理对象可以作为原有对象的“代理”,在调用原有方法时添加额外的功能或行为。这一技术在软件开发中广泛应用,特别是在实现设计模式...

    java动态替换代理ip

    Proxy类提供了创建动态代理对象的方法,而InvocationHandler接口定义了调用处理程序的接口,它处理对代理对象的所有方法调用。 动态代理的基本步骤如下: 1. 定义接口:首先,你需要定义一个或多个接口,这些接口...

    反射和动态代理(reflect and proxy)---含源码

    在Java中,有两种实现动态代理的方式:`java.lang.reflect.Proxy` 类和 `java.lang.reflect.InvocationHandler` 接口。 Proxy类是生成代理对象的工厂,而InvocationHandler接口定义了处理代理对象方法调用的逻辑。...

    java reflect and dynamic proxy .Java反射和动态代理

    1. **Proxy类**:Proxy类的newProxyInstance()方法用于创建动态代理对象,它需要一个ClassLoader、一个接口数组和一个InvocationHandler作为参数。 2. **InvocationHandler接口**:定义了代理对象在调用方法时的...

    java反射和动态代理实例

    动态代理主要由`java.lang.reflect.Proxy`和`java.lang.reflect.InvocationHandler`两个类组成。`Proxy`类用于创建代理对象,而`InvocationHandler`接口定义了处理方法调用的逻辑。我们可以通过实现`...

    java代理源码

    Java代理源码主要涉及到的是Java编程中的动态代理和网络代理技术。在Spring MVC框架下,开发者经常使用代理模式来扩展或增强对象的功能,同时,这里的"代理.war"文件表明这是一个Web应用程序,通常用于部署在如...

    Proxy & InvocationHandler

    在Java编程语言中,`Proxy`和`InvocationHandler`是两个关键的概念,它们主要用于实现动态代理,这是一种在运行时创建接口实例的技术。动态代理在很多场景下都非常有用,比如日志记录、性能监控、事务管理等。下面...

Global site tag (gtag.js) - Google Analytics