Java 虚拟机的类加载机制

关于类加载机制:

​ 虚拟机把描述类的数据从Class 文件加载到内存,并对数据进行效验、转换解析和初始化,最终 形成可以被虚拟机直接使用的Java 类型,就是虚拟机的类加载机制。

1、初始化 时机

1、遇到 new 、getstatic 、putstatic 、invokestatic 这四个字节码指令时。触发这四条指令的场景:

  • 使用new 实例化对象时
  • 读取或设置一个类的静态字段(被final修饰,已在编译期把结果放入常量池的不算(即 常量不算,static final 修饰))
  • 调用一个类的静态方法

2、使用反射调用时,先进行初始化

3、初始化一个类时,若其父类未被初始化,则先初始化其父类。

4、当虚拟机启动时,用户需要指定一个要执行的主类(main() 函数所在的类),虚拟机会先初始化它。

5、jdk 7.0 中,动态语言的支持

以上称为主动引用,被动引用不会引起初始化:

现有如下两个类:

class SuperClass{
static {
System.out.println("SuperClass init !!!");
}
public static int value = 123;
} class SubClass extends SuperClass{
static {
System.out.println("SubClass init !!!");
} public static final String HELLO_WORLG = "hello world !";
}
  • 测试一:
  public static void main(String[] args) {
System.out.println(SubClass.value);
}

运行结果

  SuperClass init !!!
123
  • 对于静态变量,只有直接定义这个变量的类才会进行初始化,如子类调用父类的静态变量,只有父类会进行初始化,子类不会自动进行初始化。

  • 测试二:

  public static void main(String[] args) {
System.out.println(SubClass.HELLO_WORLG);
}

运行结果:

  hello world !
  • 常量在在编译期通过常量传播优化,将“hello world !“存储到了常量池中,也就是说,”SubClass.HELLO_WORLG“并没有通过SubClass类符号进行引用,二者并没有任何联系。所以不会导致该类初始化。

2、加载

  • 通过类的权限定名来获取此类的二进制字节流
  • 把字节流代表的静态数据结构转化为方法区的运行时数据结构
  • 在方法区中生成一个代表这个类的Class 对象

3、验证

​ 确保class文件的字节流中的信息是安全的,至少不会危害虚拟机自身的安全。只有通过了这阶段的验证,字节流才会进入内存的方法区进行存储。

4、准备

  • 类变量分配内存并设置初始零值,这些变量的内存在方法区中进行分配。

  • 常量会设置最终值,如:

  public static int value = 123;
public static final int con = 234;

准备期过后,会把value置为0,con的值置为234。

5、解析

​ 将常量池中的符号引用替换为直接引用。这一阶段会根据需要发生在初始化之前或之后,包含类或接口解析、字段解析、方法解析。

符号引用是无关虚拟机实现的内存布局。直接引用是和虚拟机实现内存布局相关的,符号引用必须在运行期转换获得真正的内存入口地址。

6、初始化

​ 开始真正执行类中定义的 Java 代码,初始化阶段是执行类构造器() 方法的过程

  • () 是编译期收集类中所有的类变量的赋值动作静态语句块中(static{})的语句结合而成的。静态语句块只能访问惊天语句块之前的变量,定义在其之后的变量,只能赋值,不能访问
  static {
i = 0; //可以给变量赋值编译通过
System.out.println(i); //使用变量编译不通过
} static int i;
  • () 方法和构造函数不同。子类不会显示的调用父类的init() 方法,但是虚拟机会保证子类init() 方法被调用之前,父类的init() 会被先调用。
  public class InitDemo_2 {
public static void main(String[] args) {
System.out.println(SubClass1.B);
}
} class SuperClass1{
public static int A = 1;
static {
A = 2;
}
}
class SubClass1 extends SuperClass1{
public static int B = A;
}

运行结果:

  2
  • 接口中不会有静态语句块,但是接口中可以有赋值语句,因此接口也会生成() ,但是,执行接口的() 不需要先执行父类的() ,除非父类的变量被执行,才会调用父类的() 。
  • () 方法只会被执行一次

7、类加载机制

三种类加载器

  • 启动类加载器(Bootstrap ClassLoader)
​	负责加载 JAVA_HOME/lib 目录下,或被-XbootclassPath 参数指定的路径下的类库。
  • 拓展类加载器(Extension ClassLoader)

    ​ 负责加载 JAVA_HOME/lib/ext 目录下或者被 java.ext.dirs 系统变量所指定的路径中的所有类库。

  • 应用程序类加载器(Application ClassLoader)

    ​ 是ClassLoader.getSystemClassLoasder() 方法的返回值,负责加载用户类路径上所指定的类库。

双亲委派模型

  • 类加载器通过组合的方式建立的父子关系,称为双亲委派模型。

  • 类需要有加载他的类加载器和类本身一起确定其在虚拟机中的唯一性。

  • 工作流程

    ​ 一个类加载器收到了类加载加载的请求,他首先不会尝试自己加载这个类而是把这个请求委派给父类加载器来完成。只有父类无法完成这个请求时,子加载器才会尝试自己去加载。

  • 作用

    ​ Java 类随着他的类加载器一起具备了一种带优越级的层级关系,所有的加载请求都会传送到顶层的启动类加载器中,保证了Java 的稳定运行。

java 虚拟机的类加载机制的更多相关文章

  1. 深入理解java虚拟机【类加载机制】

    Java虚拟机类加载过程是把Class类文件加载到内存,并对Class文件中的数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的java类型的过程. 在加载阶段,java虚拟机需要完成以下 ...

  2. Java虚拟机:类加载机制详解

    版权声明:本文为博主原创文章,转载请注明出处,欢迎交流学习! 大家知道,我们的Java程序被编译器编译成class文件,在class文件中描述的各种信息,最终都需要加载到虚拟机内存才能运行和使用,那么 ...

  3. java虚拟机的类加载机制

    引言 我们写的代码是放在.java文件中,经过编译器编译后,转成.class文件.Class文件是一串二进制流,它可以被各平台的虚拟机所接受,实现跨平台.      虚拟机将描述类的数据从class文 ...

  4. 深入理解Java虚拟机(类加载机制)

    文章首发于微信公众号:BaronTalk 上一篇文章我们介绍了「类文件结构」,这一篇我们来看看虚拟机是如何加载类的. 我们的源代码经过编译器编译成字节码之后,最终都需要加载到虚拟机之后才能运行.虚拟机 ...

  5. 【进阶之路】深入理解Java虚拟机的类加载机制(长文)

    我们在参加面试的时候,经常被问到一些关于类加载机制的问题,也都会在面试之前准备的时候背好答案,但是我们是否有去深入了解什么是类加载机制呢?这段时间因为一些事情在家看了些书,这次就和大家分享一些关于Ja ...

  6. 深入理解Java虚拟机之类加载机制篇

    概述 ​ 虚拟机把描述类的数据从 Class 文件加载到内存中,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,就是虚拟机的类加载机制. ​ 在Java语言里面,类型的 ...

  7. Java虚拟机之类加载机制

    ⑴背景   Java虚拟机把Class文件加载到内存中,并对数据进行校验,转换解析,和初始化,最终形成被虚拟机直接使用的Java类型,这就是类加载机制. ⑵Jvm加载Class文件机制原理 类的生命周 ...

  8. 深入理解java虚拟机(三)-----类加载机制

    类加载机制jvm把描述类的数据从class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被jvm直接使用的java类型.在java中,类型的加载.连接和初始化都是在程序运行期间完成的 ...

  9. 《java虚拟机》----类加载机制

    No1: 实现语言无关性的基础仍然是虚拟机和字节码存储格式,虚拟机只与Class文件这种特定的二进制文件格式所关联,并不关心Class的来源是何种语言. No2: Class文件是一组以8位字节为基础 ...

随机推荐

  1. zoj 2022

    分析: 组合数学类型的题目. 正常的话可能会去分解1~N数里面有几个5和2,但是这样的复杂度为O(nlogn). 其实有更巧妙的办法,可以把问题分解成子问题. 可以发现N!末尾的0与1~N中有几个5的 ...

  2. 用于Mysql操作的MySqlHelper类

    using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Te ...

  3. 我的第一个python web开发框架(7)——本地部署前端访问服务器

    PS:本系列内容进度节奏会放的很慢,每次知识点都尽量少一点,这样大家接触的知识点少了,会更容易理解,因为少即是多.另外,对于后面代码部分,虽然尽量不用那些复杂的封装和类,但它并不表示看了就能全部记住, ...

  4. VUE长按事件

    PS:在开发中常常会有长按事件的需求,这里我简单的介绍几种长按事件的需求 需求一:长按数字累加或者累减 HTML: <div class="mui-numbox" data- ...

  5. [译]ASP.NET Core 2.0 带初始参数的中间件

    问题 如何在ASP.NET Core 2.0向中间件传入初始参数? 答案 在一个空项目中,创建一个POCO(Plain Old CLR Object)来保存中间件所需的参数: public class ...

  6. Mybatis基本用法--中

    Mybatis基本用法--中 第四部分 动态 SQL 动态 SQL 元素和使用 JSTL 或其他类似基于 XML 的文本处理器相似.MyBatis 采用功能强大的基于 OGNL 的表达式来消除其他元素 ...

  7. 相对定位的div没有出现纵向滚动条

    在一个相对定位的div中绝对定位很多html元素,纵向没有滚动条出现.原因:我们期望作为原点的点在窗口原点的下方,但是滚动条以窗口原点作为原点,所以没有出现滚动条.解决:在div外面再套一个div,o ...

  8. 基于HTML5 Canvas的3D动态Chart图表

    发现现在工业SCADA上或者电信网管方面用图表的特别多,虽然绝大部分人在图表制作方面用的是echarts,他确实好用,但是有些时候我们不能调用别的插件,这个时候就得自己写这些美丽的图表了,然而图表轻易 ...

  9. Numpy数组索引为-1和None

    numpy的数组操作方便,可以用:来切片,用布尔数组或者布尔表达式来查找符合条件的数据,也可以用数组作为另一个数组的索引来查找指定的数据.但有时也会见到数组索引为-1和None.两者的用法如下: 1. ...

  10. Android: Only the original thread that created a view hierarchy can touch its views 异常

    最近自己再写一个小项目练手,创建一个线程从网络获取数据然后显示在 recyclerView 上.写好后发现页面能够显示,但是有时候会把请求的数据显示过来,有时候不会.点开 android monito ...