`
lvwenwen
  • 浏览: 960876 次
  • 性别: Icon_minigender_1
  • 来自: 魔都
社区版块
存档分类
最新评论

读的JDK动态代理源码(代理类)

阅读更多
读的JDK动态代理源码(代理类) 

2010-05-21 11时51分07秒|分类: 设计模式 | 字号 订阅
读源码,怕过两天又忘记了,还是记录下吧......

动态代理最重要的实现就是Proxy.newInstance,那我们当前直接看这个方法扩展功能

(ClassLoader的加载,
       类<?> []接口,
       InvocationHandler的公共静态对象newProxyInstance  IllegalArgumentException     {  如果(H == NULL){      抛出NullPointerException异常()/ /如果InvocationHandler的为空,抛异常  }





/
  * 查找或生成指定的代理类,这个方法里最主要的地方在这里,它直接透过调用。

CL = getProxyClass(装载机,接口);

/
  * 调用指定的调用其构造  {      构造函数的利弊= cl.getConstructor(constructorParams);      cons.newInstance回报(对象)(新的Object [] {H});  }(NoSuchMethodException E){      扔:新InternalError(e.toString());  }(IllegalAccessException E){      扔新InternalError(e.toString());  渔获(InstantiationException E){      扔:新InternalError(e.toString());  }渔获(InvocationTargetException E){      扔:新InternalError(e.toString());      } }














上面的方法,除了最重要的getProxyClass,其他都很容易理解.,那么下面开始读getProxyClass方法

<> getProxyClass公共静态类(ClassLoader的装载机,
                                         <>类...接口)
抛出IllegalArgumentException
    {
如果(interfaces.length> 65535){
     抛出新IllegalArgumentException(“接口超限”); / / JDK的想的果然比较到位,连界面传动的太多都想到了..〜!〜
}

类proxyClass = NULL; / /这个就是最后要生成的二进制码,首先初始化一下

/ *收集接口名称作为键使用缓存代理类* /
的String [] interfaceNames =新的String [interfaces.length]; / /这个存放的是对应的接口的名字......

设置interfaceSet =新的HashSet()/ / 这个HashSet的是为了检测接口重复记录的。

为(int i = 0;我<interfaces.length;我+){
     /
      * 检查类加载器可以解决这个名字
      *接口的同一类对象
      * /
     字符串InterfaceName中=接口[我]的getName()。
     类interfaceClass = NULL;
     尝试{
  interfaceClass = Class.forName的(InterfaceName中,虚假,装载机); / /建立对应接口的二进制码,第二个参数虚假表示,不需要初始化
     }(ClassNotFoundException异常){
     }
     (interfaceClass =接口[I])  新IllegalArgumentException(      接口[I] +“是从类加载器中不可见”);      }



     /
      * 验证的Class对象实际上代表
      *接口。
      * /
     如果(!interfaceClass.isInterface())  新IllegalArgumentException(      interfaceClass.getName()+“不是一个接口”);      }



     /
      * 确认该接口是不是      (interfaceSet.contains(interfaceClass)){   抛出新IllegalArgumentException(       “ 反复接口:”interfaceClass.getName());      }      interfaceSet.add(interfaceClass);






     interfaceNames [I] = InterfaceName中; / /这句就是把每个接口名放到interfaceNames的数组里......

/
  * 使用
  *键缓存(而不是他们的一类代理类
  *对象)作为代理接口的字符串表示形式是足够的,因为我们需要代理
  *接口是通过提供的名称解析
  *类加载器,它具有的优点是使用一个字符串
  类*表示一个隐式的弱
  *参考类。
  * /
对象的关键=

/ *
  查找或创建代理类的类加载器缓存。
  * /
地图缓存; / /放缓存的地图
同步(loaderToCache)     (地图)loaderToCache.get“(装载机);      (缓存== NULL){   缓存=新的HashMap();   loaderToCache.put(装载机,高速缓存);      }      /       * 为这个时间      *,这种映射将继续有效方法,没有进一步的同步,因为映射      *只会被删除,如果类装载器无法访问。      * /  }











/
  * 查找接口的代理类中使用缓存列表
  *键。此查找将导致三种可能的
  *种值之一:
  *空,如果目前还没有名单
  *接口的代理类的类加载器,
  *的pendingGenerationMarker的对象,如果一个代理类
  *列表接口正在生成,
  *或弱引用类对象,如果一个代理类
  *接口名单已经产生。
  * /
同步(缓存){
     /
      * 请注意,我们不必担心收割缓存
      *清除弱引用条目,因为如果一个代理类
      *已被垃圾收集,它的类装载器将一直
      *垃圾收集一样,所以整个高速缓存将被收割
      从loaderToCache地图*。
      * /
     {
  Object值=   (参考值的instanceof){       proxyClass =(A类)((参考)  (proxyClass = NULL){       / /代理类已经产生:返回      返回proxyClass,;   }否则如果(值== pendingGenerationMarker){ / /这里的pendingGenerationMarker是一个静态常量,表示新的对象()JDK给出的解释。是,如果代理正在建立,那么等待他       / /代理类,正在生成:等待它      试图{    cache.wait();       }的catch(InterruptedException的E){    /     * 类的一代,我们正在等待为    * 应采取1小,时间有限,所以我们可以放心地忽略    *线程中断。    * /       }       继续;   }否则{       /        *已没有这个接口列表的代理类       *产生或正在产生,所以我们会去和       *生成现在标记为挂起代       * /       中cache.put(关键,pendingGenerationMarker); / /如果缓存里获取到:的对应于键的值是NULL,那么,就建立一个对象的对象放进去上面说了, pendingGenerationMarker =新的对象();       休息;        } } ,而(真);  }





























尝试{
     弦乐proxyPkg = NULL; / /包中定义代理类,这个是代理类的包名

     / *
      记录
      *代理类将在同一个包中定义的一个非公开的代理接口的包。验证
      *所有非公开的代理接口在同一个包
      *
     / (I = 0;我<interfaces.length;我+){
  INT标志。  (!Modifier.isPublic(标志)){ / /如果不是公共的接口......


      名=接口[I]的getName();
      整数n = name.lastIndexOf('。');
      弦乐PKG =((N == -1)“:name.substring(0,N + 1)); / /注意这里的N +1的,其实是包括“。”的..比如com.cjb.proxy.Proxy ..它返回的就是“com.cjb.proxy。”注意最后的那个点

      如果(proxyPkg == NULL){
   proxyPkg = PKG;
      } {(pkg.equals(proxyPkg)!)
   抛出新IllegalArgumentException(
       “ 非公开的接口,从不同的包”);
      }
 
     } }

     如果(proxyPkg == NULL){/ /如果没有非公有制代理接口,这里可以看到,如果是公共的接口,对应代理类的包名就是“”,也就是没有包名
  proxyPkg =“”; / /使用未命名的包
     }

     {
  / *
   。*选择生成代理类的名称
   * /
  长民;
  同步(nextUniqueNumberLock){
      NUM =   proxyName = proxyPkg + proxyClassNamePrefix + NUM; / / proxyPkg,什么之前产生的包名,proxyClassNamePrefix $ Proxy1美元Proxy2发出Proxy3美元一直在增长的这样的话,就避免了重复。   /    * 验证类加载器尚未   *定义类与所选择的名称。   * /






  /
   * 生成指定的代理 ,proxyClassFile = ProxyGenerator.generateProxyClass(      proxyName接口);   尝试{       proxyClass = defineClass0(装载机proxyName,   proxyClassFile,0,proxyClassFile.length);   }(ClassFormatError E){       /        * 这里一个ClassFormatError指(限制在错误       *代理类生成的代码)有一些其他       *无效方面提供的代理参数       *类的创建(如虚拟机的限制       。*超过)       * /       抛出新IllegalArgumentException(e.toString());        } }      / /添加到设置的所有生成的代理类isProxyClass,     proxyClasses.put(proxyClass,空);

 

















} {
     /
      * 我们必须清理“等候的一代”的代理状态
      *级高速缓存条目,不知何故。如果成功
      *生成一个代理类,存储在缓存中(弱引用);
      *否则,删除保留项目。在所有情况下,通知
      此缓存中保留条目*所有服务员。
      * /
     同步(缓存){
  如果(proxyClass = NULL){
      中cache.put(关键,新的WeakReference(proxyClass));
  } {
      cache.remove(键);
  }
  cache.notifyAll();
    
} }
返回proxyClass;
    }

可以看到,我们想看的,最重要的生成二进制码的方法,是native的..读了这么多源码,都是一些前期处理.关键的地方不知道..当然,虽然没看到关键的地方,但是对于它前面的处理的学习,也是非常有用的..看看JDK是怎么处理重名问题的,怎么处理cache.等等..
分享到:
评论

相关推荐

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

    动态代理类是由JDK自动生成的类,它的结构主要包括以下几个方面: 1. **代理类继承自`Proxy`类**:每一个动态代理类都是`Proxy`类的子类,这意味着它们继承了`Proxy`类中的所有属性和方法。 2. **代理类实现了目标...

    JDK动态代理源码

    这个方法接收三个参数:`ClassLoader`用于加载代理类,`Interface[]`表示代理对象需要实现的接口列表,以及`InvocationHandler`,它是处理代理对象方法调用的回调接口。 2. **InvocationHandler接口**:每个代理...

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

    在内存占用上,两者差别不大,但CGLIB生成的代理类可能略微增加内存负担。 在工具支持上,Spring框架对两者都有很好的支持。Spring AOP默认使用JDK动态代理,但如果目标类没有实现接口,它会自动切换到CGLIB。在...

    CGLIB 和 JDK生成动态代理类的区别

    性能方面,CGLIB通常比JDK动态代理更快,因为它是基于字节码生成的子类,而JDK动态代理需要创建接口的实现类。但是,如果目标类已经实现了接口,JDK动态代理的性能开销相对较小,因为它避免了子类继承带来的额外成本...

    代理模式,JDK动态代理,SpringAOP来龙去脉

    - 分析JDK动态代理的源码,可以了解其内部如何生成代理类和调用处理程序的工作机制。 - Spring AOP的源码解析有助于理解其是如何创建代理、匹配切点并执行相应通知的过程。 5. **工具应用**: - JDK动态代理和...

    JDK动态代理和Cglib动态代理实例源码

    本篇将详细探讨JDK动态代理和Cglib动态代理,并通过源码实例来加深理解。 首先,JDK动态代理是Java内置的一种机制,它依赖于`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口。`Proxy`类...

    jdk动态代理和CGlib动态代理

    1. **Proxy类**:这是JDK动态代理的核心类,它提供了`newProxyInstance()`静态方法,用于创建代理对象。 2. **InvocationHandler接口**:每个代理对象都关联一个`InvocationHandler`实例。当代理对象的方法被调用时...

    JDK8完整源码包

    源码分析可以揭示其如何实现高效调用,以及如何与Java代理(Proxy)、动态语言支持等特性相互配合。 总之,"JDK8完整源码包"是一个宝贵的学习资源,它涵盖了Java开发的核心技术和优化策略。通过深入研究,开发者不仅...

    java设计模式【之】JDK动态代理【源码】【场景:帮爸爸买菜】.rar

    java设计模式【之】JDK动态代理【源码】【场景:帮爸爸买菜】.rar /** * 代理模式 * 在开发者的角度来看,创建一个代理对象,提供给用户使用,避免用户直接访问真正的对象 * 在用户角度来看,就是普通的类方法...

    JDK动态代理在EJB3(包括WebService)中的应用

    EJB3是Java EE平台的一部分,提供了一种简化的方式来创建可部署在服务器端的企业级组件,而动态代理则允许在运行时创建代理类,以实现代理模式,通常用于拦截方法调用,进行额外的操作如日志记录、事务管理或权限...

    jdk动态代理 + 拦截器实现小例

    在Java编程中,JDK动态代理是一种非常实用的技术,它允许我们在运行时创建代理类来增强或拦截原有类的方法调用。在这个“jdk动态代理 + 拦截器实现小例”中,我们将探讨如何利用Java的InvocationHandler接口和Proxy...

    JDK的动态代理总结

    在Java开发中,JDK的动态代理是一种非常重要的技术,它允许我们在运行时创建具有特定接口的代理类实例,从而实现代理对象对目标对象的方法调用进行拦截和增强。这篇博客将对JDK的动态代理进行深入的总结,探讨其原理...

    jdk动态代理 cglib3.0动态代理

    CGLIB通过字节码技术生成代理类的实例。 1. **Enhancer类**:CGLIB的核心类,用于配置和生成代理对象。你可以设置回调方法、过滤器等,然后通过`enhance()`方法生成代理对象。 2. **MethodInterceptor接口**:类似...

    jdk11的源码src文件

    9. **jdk.dynalink**: 动态链接库允许Java代码在运行时动态地发现和调用方法。这个库在一些高级框架和库中用于实现元编程和反射增强功能。 10. **java.xml**: 这个模块包含了Java的XML处理库,包括解析器、DOM、SAX...

    JDK和CGlib分别实现的动态代理源代码

    总结一下,JDK动态代理适用于目标对象实现了接口的情况,而CGlib则适用于未实现接口的类。两者在性能上,CGlib通常比JDK代理稍快,但创建代理的过程相对复杂。选择哪种代理方式取决于具体的需求和项目情况。了解并...

    JDK动态代理demo

    `Proxy`类是用于生成动态代理类和实例的工具,而`InvocationHandler`接口定义了代理对象处理方法调用的逻辑。 首先,我们需要定义一个或多个接口,这些接口将被代理对象实现。例如,我们有一个名为`MyService`的...

    从JDK动态代理到spring AOP

    本篇将详细探讨JDK动态代理和Spring AOP,以及它们在实际应用中的作用。 首先,JDK动态代理是Java提供的一种在运行时创建代理对象的技术。它允许我们在不修改原有类的基础上,为已有接口添加额外的功能。动态代理的...

    jdk动态代理使用[文].pdf

    在Java编程中,JDK动态代理是一种强大的工具,它允许我们在运行时创建代理类来拦截对目标对象的调用,从而实现额外的功能,比如日志记录、性能监控、事务管理等,而无需修改原始代码。这里我们将深入探讨JDK动态代理...

    从JDK动态代理看Spring之AOP实现

    1. **创建代理对象**:Spring会创建一个实现了目标对象所有接口的代理类,这个代理类是Proxy类的实例。 2. **设置InvocationHandler**:Spring为代理对象设置一个实现了InvocationHandler接口的AdvisedSupport类的...

    jdk动态代理

    这是因为Java不支持多重继承,所以通过接口,代理类可以实现多个行为。 2. **InvocationHandler接口**: 在创建代理对象时,我们需要实现`java.lang.reflect.InvocationHandler`接口。这个接口定义了一个方法`...

Global site tag (gtag.js) - Google Analytics