java 类加载机制简析

java 类加载机制简析

类的加载:指的是 JVM 将类的 .class 文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个这个类的 java.lang.Class 对象,用来封装类在方法区类的对象,加载的最终目标是堆中的 Class 对象

Class 对象封装了类在方法区内的数据结构,并且向 Java 程序员提供了访问方法区内的数据结构的接口

JVM 将类加载过程分为三个步骤:装载(Load)链接(Link)初始化(Initialize)

classload.gif

类装载(不同加载器的主要不同点)

来源(.class文件来源):

  • 从本地系统直接加载.class
  • 网络下载.class
  • zip,jar 等归档文件中加载.class
  • 从专有数据库中提取.class
  • 将 Java 源文件动态编译为 .class 文件(服务器)

类链接

  1. 验证:确保被加载类的正确性;

  2. 准备:为类的静态变量分配内存,并将其初始化为默认值;

  3. 解析:把类中的符号引用转换为直接引用;

类初始化

  • WHEN:

    1. 创建类的实例,也就是 new 一个对象
    2. 访问某个类或接口的静态变量,或者对该静态变量赋值
    3. 调用类的静态方法
    4. 反射Class.forName()
    5. 初始化一个类的子类(会首先初始化子类的父类)
    6. JVM 启动时标明的启动类,即文件名和类名相同的那个类
  • HOW:

    1. 如果这个类还没有被装载和链接,那先进行装载和链接

    2. 假如这个类存在直接父类,并且这个类还没有被初始化(注意:在一个类加载器中,类只能初始化一次),那就初始化直接的父类(不适用于接口)

    3. 加入类中存在初始化语句(如static变量和static块),那就依次执行这些初始化语句

load-init.PNG
load-init order.PNG
 /**
 * 初始化伪代码
 */
 init (Class child) {
   if (child.hasParent()) {
     Class<?> parent = child.getParent();
     loadAndLink(parent);
     init(parent);
   }
   /**
   * 初始化静态成员代码。。。。
   */
   return;
 }
 

 ​

类加载器

JVM 的类加载是通过 ClassLoader 及其子类来完成

  • WHAT:实际上,各种类加载器其实主要对 .class 源进行操作,并将其转化成字节序列
  • WHY:在 JVM 中,一个类由其全限定名和一个加载类 ClassLoader 的实例作为唯一标识,不同类加载器加载的类将被置于不同的命名空间
  • HOW:组成了如下图所示的 ,以 ”双亲委托机制“为”桥梁“的 ClassLoader Architecture

注意:

BootstrapClassLoader: 由 c++ 编写而成,不是ClassLoader的子类

ExtClassLoaderAppClassLoader 没有类继承关系,却通过 parent 属性使其构成父子类加载器的关系

classloader.gif

上图表示出 java 类加载机制中的 “双亲委托机制“

双亲委托机制

  • WHAT:当前类加载器在调用 loadClass() 加载类时,主要进行下步骤:

    1. 检查类是否已被加载过,调用 findClassLoaded() 查看当前类加载器是否存在 class 的缓存
    2. 若类未被加载过,递归委托父类加载器调用 loadClass() 加载类,若无,则 findBootstrapClassOrNull() 完成类加载
    3. 若以上步骤都不能完成类加载,则调用 findClass() 尝试当前类加载器完成加载,若加载成功则缓存

    特点:

    1. 从子到父共享缓存
    2. 从父到子尝试自加载
  • WHY:双亲委托机制主要是为了 ”类加载器之间共享缓存“"类加载器的优先级问题",保证基础类的类加载器统一的问题

  • HOW:具体代码实现:

    /**
    * //:ClassLoader 类中的 loadClass() 源码
    * loadClass 是加载类的入口,也是双亲委托机制的实现
    * @param name 必须为二进制名,即类的全限定名
    * @param resolve 是否链接类,链接绑定类与加载器
    * @return 类加载完成后返回存储在堆中 Class
    * 方法 loadClass 并未被 final 修饰,代表可以被复写,从而破坏“双亲委托”
    */
    protected Class<?> loadClass(String name, boolean resolve)
           throws ClassNotFoundException
       {
           synchronized (getClassLoadingLock(name)) {
               // 检查类是否已被加载过
               Class<?> c = findLoadedClass(name);
               if (c == null) {
                   long t0 = System.nanoTime();
                   try {
                       if (parent != null) {
                         //递归委托父类加载器加载类
                           c = parent.loadClass(name, false);
                       } else {
                         //无父类加载器,则调用启动类加载器加载
                           c = findBootstrapClassOrNull(name);
                       }
                   } catch (ClassNotFoundException e) {
                       // ClassNotFoundException thrown if class not found
                       // from the non-null parent class loader
                   }
      
                   if (c == null) {
                       long t1 = System.nanoTime();
                      //递归委托后仍然无法完成类加载,则使用当前类加载器加载
                       c = findClass(name);
      
                       // this is the defining class loader; record the stats
                       sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                       sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                       sun.misc.PerfCounter.getFindClasses().increment();
                   }
               }
               if (resolve) {
                   resolveClass(c);
               }
               return c;
           }
       }
    

自定义类加载器(extends ClassLoader)的方式:

  • override loadClass() 破化双亲委托机制(不推荐)
  • override findClass() 兼容双亲委托机制,通过复写提供的 findClass() 来实现在双亲委托机制的基础上来自定义类加载器
    • 先内后外,加载类时,遵循先双亲委托,后自定义加载

ClassLoader 的三个构造器:

    
    private final ClassLoader parent; //父类加载器,只在构造器中一次性指定,不提供public方法设置
    /**
    * ClassLoader 无参构造器
    * 内部调用带参构造器,使用getSystemClassLoader()作为parent形参的实参
    * getSystemClassLoader()返回的是sun.misc.Launcher$AppClassLoader
    */
    protected ClassLoader() {
          this(checkCreateClassLoader(), getSystemClassLoader());
      }

    /**
    * 带参构造器
    * @parent 用来指定父类加载器
    */
    protected ClassLoader(ClassLoader parent) {
        this(checkCreateClassLoader(), parent);
    }

    /**
    * 带参构造器
    */
    private ClassLoader(Void unused, ClassLoader parent) {
          this.parent = parent;  //设置父类加载器
          if (ParallelLoaders.isRegistered(this.getClass())) {
              parallelLockMap = new ConcurrentHashMap<>();
              package2certs = new ConcurrentHashMap<>();
              domains =
                  Collections.synchronizedSet(new HashSet<ProtectionDomain>());
              assertionLock = new Object();
          } else {
              // no finer-grained lock; lock on the classloader instance
              parallelLockMap = null;
              package2certs = new Hashtable<>();
              domains = new HashSet<>();
              assertionLock = this;
          }
      }

实现自定类加载器:

      /**
      * //: findClass 方法源码
      * @name 
      * 该方法只抛出一个 ClassNotFountException(name)
      * 表明此方法被设计来复写,作为自定义加载器类的接口
      */
    protected Class<?> findClass(String name) throws ClassNotFoundException {
          throw new ClassNotFoundException(name);
        // 自定义时,在此处添加代码,将 .class 源解析为 byte[] classData
        //  return defineClass(name, classData, 0, classData.length);  
      }
    /**
    * //: defindClass 方法源码
    * final 修饰,不能被子类复写,提供真正意义上 JVM 对类进行加载的接口
    * 将由.class 解析成的字节序列,存储于方法区,然后在堆中生成对应的Class对象
    */
    protected final Class<?> defineClass(String name, byte[] b, int off, int len)
        throws ClassFormatError
    {
        return defineClass(name, b, off, len, null);
    }
                        ___$$$$$$$_____________________________
________________________$$$$$$$$$$___________________________
________________________$$$$$$$$$$$__________________________
_________________________$$$$$$$$$$$$$$______________________
__________________________$$$$$$$$$$$________________________
_____________________________$$$$$$$$$$$$$___________________
___________________________$$$$$$$$$$________________________
_________________________$$$$$$$$$$$$$$$_____________________
________________$$$______$$$$$$$$$$$$$$______________________
______________$$$$$$$$_____$$$$$$__$$$$$_____________________
_____________$$$$$$$$$$_____$$$$____$$$$$____________________
___________$$$$$$_$$$$$$$$__$$$$______$$$$___________________
__________$$$$$_____$$$$$$$$_$$$$_______$$$__________________
________$$$$$_________$$$$$$$$$$$$_______$$$_________________
_______$$$_____________$$$$$$$$$$$________$$$________________
_____$$$________________$$$$$$$$$$________$$$$$$_____________
__$$$$$$__________________$$$$$$$
-------------------------------------------------------------
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,723评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,485评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,998评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,323评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,355评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,079评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,389评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,019评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,519评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,971评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,100评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,738评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,293评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,289评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,517评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,547评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,834评论 2 345

推荐阅读更多精彩内容