JVM系列—Class文件加载过程

开发 后端
虚拟机的基础概念class文件结构class文件加载过程jvm内存模型JVM常用指令GC与调优,JVM加载Class文件主要分3个过程:Loading 、Linking、Initialzing

[[347016]]

 JVM系列笔记目录
虚拟机的基础概念class文件结构class文件加载过程jvm内存模型JVM常用指令GC与调优

Class文件加载过程
JVM加载Class文件主要分3个过程:Loading 、Linking、Initialzing

1.Loading
Loading的过程就是通过类加载器将.class文件加载到jvm内存中过程。需要理解双亲委派机制、类加载器ClassLoader,加载过程如下。

ClassLoader

不同的类加载器加载范围不一样,以Java8中的为例。

BootClassLoader 加载范围sun.boot.class.pahtExtClassLoader 加载范围java.ext.dirsAppClassLoader 加载范围java.class.pathCustomClassLoader 可自定义加载范围

前三个加载器来自JDK的Launcher类,三个ClassLoader作为Launcher的内部类,感兴趣可以查看下源码。

开发者也可以自定义的ClassLoader,自定义记载范围。

双亲委派机制

自底向上检查该类是否已经加载,parent方向;自顶向下进行类的实际查找和加载,child方向。 类的加载遵循双亲委派机制,主要是出于安全的考虑。双亲委派机制是如何实现的,下面源码会解释。

​ 注意:双亲委派中存在所谓的父加载器并不是加载器的加载器,只是翻译的问题,别混淆了类的继承概念。

ClassLoader源码

ClassLoader源码中比较重要的一个函数是loadClass(),执行过程是:findLoadedClass()->parrent.loadClass()->findClass(),第一步是自底向上查询是否已经加载,第二步是自顶向下查找加载类。这里就规定或是说实现了双亲委派机制。详细见ClassLoader的源码。

自定义ClassLoader

如何自定义ClassLoader?可以继承ClassLoader类,重新自己的findClass(),在里面调用defineClass()来实现自定义加载特定范围的类。

如何打破双亲委派机制,哪种情形下打破过?

从上面的ClassLoader源码中大概能看出是如何实现了双亲委派机制的,从这入手可以通过2种方式打破该机制:

super(parent)指定parent会打破该机制自定义ClassLoader重写loadClass()也可以打破

何时打破过?双亲委派机制并不是不能打破,某些特殊场景下也会选择打破该机制。

JDK 1.2之前,自定义ClassLoader必须重写loadClass(),打破过。线程ThreadContextClassLoader可以实现基础类调用实现类代码,通过thread.setContextClassLoader指定。热启动热部署,如tomcat都有自己模块指定的classloader,可以加载同一类库的不同版本。

Class执行方式

Class执行方式分为3种:解释执行、编译执行、混合执行,各有优缺点,可通过参数指定。

1.解释执行:使用bytecode intepreter 解释器解释执行,该模式启动很快,执行稍慢,可通过-Xint参数指定该模式。
2.编译执行:使用 Just in time Complier JIT编译器编译执行,该模式执行很快,编译很慢,可通过-Xcomp参数指定该模式。
3.混合执行:默认的模式,解释器+热点代码编译,开始解释执行,启动较快,对热点代码进行实时监测和编译成本地代码执行,可通过-Xmixed参数指定该模式。
热点代码监测:多次被调用的方法用方法计数器,多次被调用的循环用循环计数器,可通过参数-XX:CompileThreshold = 10000指定触发JIT编译的阈值。

2.Linking
Linking链接的过程分3个阶段:Vertification、Preparation、Resolution。

  • Vertification: 验证Class文件是否符合JVM规定。
  • Preparation:给静态成员变量赋默认值
  • Resolution:将类、方法、属性等符号引用解释为直接引用;常量池中的各种符号引用解释为指针、偏移量等内存地址的直接引用

3. Initializing
调用初始化代码clint,给静态成员变量赋初始值。

这里可以了解下必须初始化的5种情况:

new getstatic putstatic invokestatic指令,访问final变量除外java.lang.reflect对类进行反射调用时初始化子类的时候,父类必须初始化虚拟机启动时,被执行的主类必须初始化动态语言支持java.lang.invoke.MethodHandler解释的结果为REF_getstatic REF_putstatic REF_invokestatic的方法句柄时,该类必须初始化。

4.总结思考
设计模式中单例模式的双重检查的实现,INSTANCE是否需要加valatile?

  1. public class Mgr06 { 
  2.     // 是否需要加volatile? 
  3.     private static volatile Mgr06 INSTANCE; 
  4.  
  5.     private Mgr06() { 
  6.     } 
  7.  
  8.     public static Mgr06 getInstance() { 
  9.         if (INSTANCE == null) { 
  10.             //双重检查 
  11.             synchronized (Mgr06.class) { 
  12.                 if(INSTANCE == null) { 
  13.                     try { 
  14.                         Thread.sleep(1); 
  15.                     } catch (InterruptedException e) { 
  16.                         e.printStackTrace(); 
  17.                     } 
  18.                     // new 了对象,不为null,但未完成变量的初始化复制,对象处于半初始化状                    态,其它线程有可能取到半初始化的对象。 
  19.                     INSTANCE = new Mgr06(); 
  20.                 } 
  21.             } 
  22.         } 
  23.         return INSTANCE; 
  24.     } 
  25. 复制代码 

个人认为是需要加的。思考方向, class文件load到内存,给静态变量赋默认值,再赋初始值,new 对象的时候,首先要申请内存空间,然后给成员变量赋默认值,接下来给成员变量赋初始值,这个过程中对象有可能处于半初始化状态,多线程并发下别的线程有可能取到半初始化的对象,加volatile可保证线程的可见性。

责任编辑:姜华 来源: 今日头条
相关推荐

2020-11-17 09:55:48

Java

2012-03-01 10:51:37

JavaJVM

2017-02-07 09:54:43

JVMJavaClass

2017-09-20 08:07:32

java加载机制

2024-08-09 11:50:00

2009-07-08 11:01:40

Servlet源文件

2022-01-14 11:45:40

JVM 虚拟机Java

2020-12-21 08:11:46

JVMJDKJRE

2017-09-20 08:48:09

JVM内存结构

2009-02-03 09:42:53

JAVA类JVM指令forName方法

2020-07-10 16:00:57

dumpJava内存

2023-10-31 16:00:51

类加载机制Java

2017-10-17 14:02:30

jvm调优工具

2017-09-22 15:15:23

jvm调优命令

2017-11-01 15:38:54

jvm知识点总览

2024-03-28 12:32:18

JVM类加载构造器

2011-11-18 11:04:03

Java框架Equinox

2023-08-02 08:38:27

JVM加载机制

2021-10-11 19:54:04

JVM面试虚拟机

2021-02-28 11:58:33

JVM机制语言
点赞
收藏

51CTO技术栈公众号