Java面试中最常问的JVM问题大全,附答案!

​本篇文章总结了目前JVM面试中最常问到的问题,如下是问题,答案附在后面java


Java 内存分配程序员

简述 Java 垃圾回收机制web

垃圾回收的优势和原理并考虑 2 种回收机制面试

System.gc() Runtime.gc() 会作什么事情?算法

Java 堆的结构是什么样子的?什么是堆中的永久代(Perm Gen space)****?编程

Java 中会存在内存泄漏吗,请简单描述数组

Java 中的内存泄露的状况缓存

finalize() 方法何时被调用?析构函数 (finalization) 的目的是什么?多线程

JVM 的永久代中会发生垃圾回收么?分布式

什么是类加载器,类加载器有哪些?

Java 类加载过程?

类加载的主要步骤?


Java 内存分配

• 寄存器:程序计数器,是线程私有的,就是一个指针,指向方法区中的方法字节码。

• 静态域:static 定义的静态成员。

• 常量池:编译时被肯定并保存在 .class 文件中的(final)

常量值和一些文本修饰的符号引用(类和接口的全限定名,字段的名称和描述符,方法和名称和描述符)。

• 非 RAM 存储:硬盘等永久存储空间。

• 堆内存:new 建立的对象和数组,由 Java 虚拟机自动垃圾回收器管理,存取速度慢。

• 栈内存:基本类型的变量和对象的引用变量(堆内存空间的访问地址),速度快,能够共享,可是大小与生存期必须肯定,缺少灵活性。

串行(serial)收集器和吞吐量(throughput)收集器的区别是什么?

吞吐量收集器使用并行版本的新生代垃圾收集器,它用于中等规模和大规模数据的应用程序。

而串行收集器对大多数的小应用(在现代处理器上须要大概 100M 左右的内存)就足够了。

在 Java 中,对象何时能够被垃圾回收?

当对象对当前使用这个对象的应用程序变得不可触及的时候,这个对象就能够被回收了。

GC 是什么? 为何要有 GC?

GC 是垃圾收集的意思(GabageCollection),内存处理是编程人员容易出现问题的地方,忘记或者错误的内存回收会致使程序或

系统的不稳定甚至崩溃,Java 提供的 GC 功能能够自动监测对象是否超过做用域从而达到自动回收内存的目的,Java 语言没有提

供释放已分配内存的显示操做方法。

简述 Java 垃圾回收机制

在 Java 中,程序员是不须要显示的去释放一个对象的内存的,而是由虚拟机自行执行。在 JVM 中,有一个垃圾回收线程,它是低优先级的,在正常状况下是不会执行的,只有在虚拟机空闲或者当前堆内存不足时,才会触发执行,扫面那些没有被任何引用的对象,并将它们添加到要回收的集合中,进行回收。

如何判断一个对象是否存活?(或者 GC 对象的断定方法)

判断一个对象是否存活有两种方法:

引用计数法

所谓引用计数法就是给每个对象设置一个引用计数器,每当有一个地方引用这个对象时,就将计数器加一,引用失效时,计数器就

减一。当一个对象的引用计数器为零时,说明此对象没有被引用,也就是“死对象”,将会被垃圾回收.引用计数法有一个缺陷就是没法解决循环引用问题,也就是说当对象 A 引用对象 B,对象 B 又引用者对象 A,那么此时 A、B 对象的引用计数器都不为零,也就形成没法完成垃圾回收,因此主流的虚拟机都没有采用这种算法。

可达性算法(引用链法)

该算法的思想是:从一个被称为 GC Roots 的对象开始向下搜索,若是一个对象到 GC Roots 没有任何引用链相连时,则说明此对

象不可用。

在 Java 中能够做为 GC Roots 的对象有如下几种:

• 虚拟机栈中引用的对象

• 方法区类静态属性引用的对象

• 方法区常量池引用的对象

• 本地方法栈 JNI 引用的对象

虽然这些算法能够断定一个对象是否能被回收,可是当知足上述条件时,一个对象并不必定会被回收。当一个对象不可达 GC Root

时,这个对象并不会立马被回收,而是处于一个死缓的阶段,若要被真正的回收须要经历两次标记.若是对象在可达性分析中没有与 GC Root 的引用链,那么此时就会被第一次标记而且进行一次筛选,筛选的条件是是否有必要执行finalize() 方法。当对象没有覆盖 finalize() 方法或者已被虚拟机调用过,那么就认为是不必的。若是该对象有必要执行finalize() 方法,那么这个对象将会放在一个称为 F-Queue 的对

队列中,虚拟机会触发一个 Finalize() 线程去执行,此线程是低优先级的,而且虚拟机不会承诺一直等待它运行完,这是由于若是finalize() 执行缓慢或者发生了死锁,那么就会形成 F-Queue 队列一直等待,形成了内存回收系统的崩溃。GC 对处于 F-Queue 中的对象进行第二次被标记,这时,该对象将被移除” 即将回收”集合,等待回收。

垃圾回收的优势和原理并考虑 2 种回收机制

Java 语言中一个显著的特色就是引入了垃圾回收机制,使 C++ 程序员最头疼的内存管理的问题迎刃而解,它使得 Java 程序员在编写程序的时候再也不须要考虑内存管理。因为有个垃圾回收机制,Java 中的对象再也不有“做用域”的概念,只有对象的引用才有"做用域"。垃圾回收能够有效的防止内存泄露,有效的使用可使用的内存。垃圾回收器一般是做为一个单独的低级别的线程运行,不可预知的状况下对内存堆中已经死亡的或者长时间没有使用的对象进行清楚和回收,程序员不能实时的调用垃圾回收器对某个对象或全部对象进行垃圾回收。

回收机制有分代复制垃圾回收和标记垃圾回收,增量垃圾回收。

垃圾回收器的基本原理是什么?垃圾回收器能够立刻回收内存吗?有什么办法主动通知虚拟机进行垃圾回收?

对于 GC 来讲,当程序员建立对象时,GC 就开始监控这个对象的地址、大小以及使用状况。一般,GC 采用有向图的方式记录和管理堆(heap)中的全部对象。经过这种方式肯定哪些对象是”可达的”,哪些对象是”不可达的”。当 GC 肯定一些对象为“不可达”时,GC 就有责任回收这些内存空间。能够。程序员能够手动执行 System.gc(),通知 GC 运行,可是 Java 语言规范并不保证 GC 必定会执行。

System.gc() Runtime.gc() 会作什么事情?

这两个方法用来提示 JVM 要进行垃圾回收。可是,当即开始仍是延迟进行垃圾回收是取决于 JVM 的。

Java 堆的结构是什么样子的?什么是堆中的永久代(Perm Gen space)****?

JVM 的堆是运行时数据区,全部类的实例和数组都是在堆上分配内存。它在 JVM 启动的时候被建立。对象所占的堆内存是由自动

内存管理系统也就是垃圾收集器回收。堆内存是由存活和死亡的对象组成的。存活的对象是应用能够访问的,不会被垃圾回收。死亡的对象是应用不可访问尚且尚未被垃圾收集器回收掉的对象。一直到垃圾收集器把这些 对象回收掉以前,他们会一直占据堆内存空间。

Java 中会存在内存泄漏吗,请简单描述。

所谓内存泄露就是指一个再也不被程序使用的对象或变量一直被占据在内存中。Java 中有垃圾回收机制,它能够保证一对象再也不被引用的时候,即对象变成了孤儿的时候,对象将自动被垃圾回收器从内存中清除掉。因为 Java 使用有向图的方式进行垃圾回收管理,能够消除引用循环的问题,例若有两个对象,相互引用,只要它们和根进程不可达的,那么 GC 也是能够回收它们的,

下面的代码能够看到这种状况的内存回收:

import java.io.IOException; **public**  **class**  **GarbageTest** {**public**  **static**  **void**  **main**(String[] args) throws IOException {
 // TODO Auto-generated method stub **try** {
 gcTest();
 } **catch** (IOException e) {// TODO Auto-generated catch block e.printStackTrace();
 }
 System.**out**.println("has exited gcTest!");
 System.**in**.read();
 System.**in**.read();
 System.**out**.println("out begin gc!");
 **for**(**int** i=0;i<100;i++){
 System.gc();
 System.**in**.read();
 System.**in**.read();
 }
 }
 **private**  **static**  **void**  **gcTest**() throws IOException {
 System.**in**.read();
 System.**in**.read();
 Person p1 = **new** Person();
 System.**in**.read();
 System.**in**.read();
 Person p2 = **new** Person();
 p1.setMate(p2);
 p2.setMate(p1);
 System.**out**.println("before exit gctest!");
 System.**in**.read();
 System.**in**.read();
 System.gc();
 System.**out**.println("exit gctest!");
 }
 **private**  **static**  **class**  **Person**{ **byte**[] data = **new**  **byte**[20000000];
 Person mate = null;
 **public**  **void**  **setMate**(Person other){
 mate = other;
 }
 }
}

Java 中的内存泄露的状况:

长生命周期的对象持有短生命周期对象的引用就极可能发生内存泄露,尽管短生命周期对象已经再也不须要,可是由于长生命周期对象持有它的引用而致使不能被回收,这就是 Java 中内存泄露的发生场景,通俗地说,就是程序员可能建立了一个对象,之后一直再也不使用这个对象,这个对象却一直被引用,即这个对象无用可是却没法被垃圾回收器回收的,这就是 java 中可能出现内存泄露的状况,例如,缓存系统,咱们加载了一个对象放在缓存中 (例如放在一个全局 map 对象中),而后一直再也不使用它,这个对象一直被缓存引用,但却再也不被使用。检查 Java 中的内存泄露,必定要让程序将各类分支状况都完整执行到程序结束,而后看某个对象是否被使用过,若是没有,则才能

断定这个对象属于内存泄露。若是一个外部类的实例对象的方法返回了一个内部类的实例对象,这个内部类对象被长期引用了,即便那个外部类实例对象再也不被使用,但因为内部类持久外部类的实例对象,这个外部类对象将不会被垃圾回收,这也会形成内存泄露。

**public**  **class** Stack {**private** Object[] elements=**new** Object[10];
 **private**  **int**  **size** = 0;
 **public**  **void**  **push**(Object e){
 ensureCapacity();
 elements[**size**++] = e;
 }
 **public** Object **pop**(){
 **if**( **size** == 0) **throw**  **new** EmptyStackException();**return** elements[--**size**];
 }
 **private**  **void** ensureCapacity(){**if**(elements.length == **size**){
 Object[] oldElements = elements;
 elements = **new** Object[2 * elements.length+1];
 System.arraycopy(oldElements,0, elements, 0, **size**);
 }
 } }

上面的原理应该很简单,假如堆栈加了 10 个元素,而后所有弹出来,虽然堆栈是空的,没有咱们要的东西,可是这是个对象是没法回收的,这个才符合了内存泄露的两个条件:无用,没法回收。可是就是存在这样的东西也不必定会致使什么样的后果,若是这个

堆栈用的比较少,也就浪费了几个 K 内存而已,反正咱们的内存都上 G 了,哪里会有什么影响,再说这个东西很快就会被回收的,

有什么关系。下面看两个例子。

public class Bad{ public static Stack s=Stack();static{
 s.push(new Object());
 s.pop(); //这里有一个对象发生内存泄露 s.push(new Object()); //上面的对象能够被回收了,等因而自 愈了 } }

由于是 static,就一直存在到程序退出,可是咱们也能够看到它有自愈功能,就是说若是你的 Stack 最多有 100 个对象,那么最

多也就只有 100 个对象没法被回收其实这个应该很容易理解,Stack 内部持有 100 个引用,最坏的状况就是他们都是无用的,

由于咱们一旦放新的进去,之前的引用天然消失!内存泄露的另一种状况:当一个对象被存储进 HashSet 集合中之后,就不能修改这对象中的那些参与计算哈希值的字段了,不然,对象修改后的哈希值与最初存储进 HashSet 集合中时的哈希值就不一样了,在这种状况下,即便在 contains 方法使用该对象的当前引用做为的参数去 HashSet 集合中检索对象,也将返回找不到对象的结果,这也会致使没法从HashSet 集合中单独删除当前对象,形成内存泄露。

深拷贝和浅拷贝。

简单来说就是复制、克隆。

Person p=new Person(“张三”);

浅拷贝就是对对象中的数据成员进行简单赋值,若是存在动态成员或者指针就会报错。深拷贝就是对对象中存在的动态成员或指针从新开辟内存空间。

finalize() 方法何时被调用?析构函数 (finalization) 的目的是什么?

垃圾回收器(garbage colector)决定回收某对象时,就会运行该对象的 finalize() 方法 可是在 Java 中很不幸,若是内存老是充

足的,那么垃圾回收可能永远不会进行,也就是说 filalize() 可能永远不被执行,显然期望它作收尾工做是靠不住的。

那么****finalize() 到底是作什么的呢?

它最主要的用途是回收特殊渠道申请的内存。Java 程序有垃圾回收器,因此通常状况下内存问题不用程序员操心。但有一种 JNI(Java Native Interface)调用non-Java 程序(C 或 C++), finalize() 的工做就是回收这部分的内存。

若是对象的引用被置为 null**,垃圾收集器是否会当即释放对象占用的内存?**

不会,在下一个垃圾回收周期中,这个对象将是可被回收的。

什么是分布式垃圾回收(DGC)?它是如何工做的?

DGC 叫作分布式垃圾回收。RMI 使用 DGC 来作自动垃圾回收。由于 RMI 包含了跨虚拟机的远程对象的引用,垃圾回收是很困难的。DGC 使用引用计数算法来给远程对象提供自动内存管理。

简述 Java 内存分配与回收策率以及 Minor GC Major GC**。**

• 对象优先在堆的 Eden 区分配

• 大对象直接进入老年代

• 长期存活的对象将直接进入老年代

当 Eden 区没有足够的空间进行分配时,虚拟机会执行一次Minor GC。Minor GC 一般发生在新生代的 Eden 区,在这个区的对象生存期短,每每发生 Gc 的频率较高,回收速度比较快;

Full GC/Major GC 发生在老年代,通常状况下,触发老年代 GC 的时候不会触发 Minor GC,可是经过配置,能够在 Full GC 之

前进行一次 Minor GC 这样能够加快老年代的回收速度。

JVM 的永久代中会发生垃圾回收么?

垃圾回收不会发生在永久代,若是永久代满了或者是超过了临界值,

会触发彻底垃圾回收(Full GC)。

注:Java 8 中已经移除了永久代,新加了一个叫作元数据区的

native 内存区。

Java 中垃圾收集的方法有哪些?

标记 - 清除:这是垃圾收集算法中最基础的,根据名字就能够知

道,它的思想就是标记哪些要被回收的对象,而后统一回收。这种

方法很简单,可是会有两个主要问题:

  1. 效率不高,标记和清除的效率都很低;

  2. 会产生大量不连续的内存碎片,致使之后程序在分配较大的 对象时,因为没有充足的连续内存而提早触发一次 GC 动做。

复制算法:

为了解决效率问题,复制算法将可用内存按容量划分为相等的两部分,而后每次只使用其中的一块,当一块内存用完时,就将还存活的对象复制到第二块内存上,而后一次性清楚完第一块内存,再将第二块上的对象复制到第一块。可是这种方式,内存的代价过高,每次基本上都要浪费通常的内存。因而将该算法进行了改进,内存区域再也不是按照 1:1 去划分,而是将内存划分为 8:1:1 三部分,较大那分内存交 Eden 区,其他是两块较小的内存区叫 Survior 区。每次都会优先使用 Eden 区,若 Eden 区满,就将对象复制到第二块内存区上,而后清除 Eden区,若是此时存活的对象太多,以致于 Survivor 不够时,会将这些对象经过分配担保机制复制到老年代中。(java 堆又分为新生代和老年代)

标记 - 整理:

该算法主要是为了解决标记 - 清除,产生大量内存碎片的问题;当对象存活率较高时,也解决了复制算法的效率问题。它的不一样之处就是在清除对象的时候现将可回收对象移动到一端,而后清除掉端边界之外的对象,这样就不会产生内存碎片了。

分代收集:

如今的虚拟机垃圾收集大多采用这种方式,它根据对象的生存周期,将堆分为新生代和老年代。在新生代中,因为对象生存期短,每次回收都会有大量对象死去,那么这时就采用复制算法。老年代里的对象存活率较高,没有额外的空间进行分配担保。

什么是类加载器,类加载器有哪些?

实现经过类的权限定名获取该类的二进制字节流的代码块叫作类加载器。

主要有如下四种类加载器:

• 启动类加载器(Bootstrap ClassLoader)用来加载 Java 核心类库,没法被 Java 程序直接引用。

• 扩展类加载器(extensions class loader):它用来加载 Java 的扩展库。Java 虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载 Java 类。

• 系统类加载器(system class loader):它根据 Java 应用的类路径(CLASSPATH)来加载 Java 类。通常来讲,Java 应用的类都是由它来完成加载的。能够经过ClassLoader.getSystemClassLoader() 来获取它。

• 用户自定义类加载器,经过继承 java.lang.ClassLoader 类的方式实现。

类加载器双亲委派模型机制?

当一个类收到了类加载请求时,不会本身先去加载这个类,而是将其委派给父类,由父类去加载,若是此时父类不能加载,反馈给子类,由子类去完成类的加载。

1. 内存模型以及分区,须要详细到每一个区放什么。

JVM 分为堆区和栈区,还有方法区,初始化的对象放在堆里面,引用放在栈里面,class 类信息常量池(static 常量和 static 变量)等放在方法区

· 方法区:主要是存储类信息,常量池(static 常量和 static 变量),编译后的代码(字节码)等数据

· 堆:初始化的对象,成员变量 (那种非 static 的变量),全部的对象实例和数组都要在堆上分配

· 栈:栈的结构是栈帧组成的,调用一个方法就压入一帧,帧上面存储局部变量表,操做数栈,方法出

· 本地方法栈:主要为 Native 方法服务

· 程序计数器:记录当前线程执行的行号

2. 堆里面的分区:Eden,survival (from+ to),老年代,各自的特色。

堆里面分为新生代和老生代(java8 取消了永久代,采用了 Metaspace),新生代包含 Eden+Survivor 区,survivor 区里面分为 from 和 to 区,内存回收时,若是用的是复制算法,从 from 复制到 to,当通过一次或者屡次 GC 以后,存活下来的对象会被移动到老年区,当 JVM 内存不够用的时候,会触发 Full GC,清理 JVM 老年区当新生区满了以后会触发 YGC,先把存活的对象放到其中一个 Survice区,而后进行垃圾清理。由于若是仅仅清理须要删除的对象,这样会致使内存碎片,所以通常会把 Eden 进行彻底的清理,而后整理内存。那么下次 GC 的时候,就会使用下一个 Survive,这样循环使用。若是有特别大的对象,新生代放不下,就会使用老年代的担保,直接放到老年代里面。由于 JVM 认为,通常大对象的存活时间通常比较久远。

3. 对象建立方法,对象的内存分配,对象的访问定位。new 一个对象

4. GC 的两种断定方法:

引用计数法:指的是若是某个地方引用了这个对象就+1,若是失效了就-1,当为 0 就会回收可是 JVM 没有用这种方式,由于没法断定相互循环引用(A 引用 B,B 引用 A)的状况

引用链法:经过一种 GC ROOT 的对象(方法区中静态变量引用的对象等-static 变量)来判断,若是有一条链可以到达 GC ROOT 就说明,不能到达 GC ROOT 就说明能够回收

SafePoint 是什么

好比 GC 的时候必需要等到 Java 线程都进入到 safepoint 的时候 VMThread 才能开始执行 GC,

  1. 循环的末尾 (防止大循环的时候一直不进入 safepoint,而其余线程在等待它进入safepoint)

  2. 方法返回前

  3. 调用方法的 call 以后

  4. 抛出异常的位置

6. GC 的三种收集方法:标记清除、标记整理、复制算法的原理与特色,分别用在什么地方,若是让你优化收集方法,有什么思路?

先标记,标记完毕以后再清除,效率不高,会产生碎片复制算法:分为 8:1 的 Eden 区和 survivor 区,就是上面谈到的 YGC标记整理:标记完毕以后,让全部存活的对象向一端移动

GC 收集器有哪些?CMS 收集器与 G1 收集器的特色。

并行收集器:串行收集器使用一个单独的线程进行收集,GC 时服务有停顿时间

串行收集器:次要回收中使用多线程来执行CMS 收集器是基于“标记—清除”算法实现的,通过屡次标记才会被清除

G1 从总体来看是基于“标记—整理”算法实现的收集器,从局部(两个 Region 之间)上来看是基于“复制”算法实现的

Minor GC 与 Full GC 分别在何时发生

新生代内存不够用时候发生 MGC 也叫 YGC,JVM 内存不够的时候发生 FGC

几种经常使用的内存调试工具:jmap、jstack、jconsole、jhat

jstack 能够看当前栈的状况,jmap 查看内存,jhat 进行 dump 堆的信息

简述 java 内存分配与回收策率以及 Minor GC 和

Major GC

  1. 对象优先在堆的 Eden 区分配。

  2. 大对象直接进入老年代.

  3. 长期存活的对象将直接进入老年代. 当 Eden 区没有足够的空间进行分配时,虚拟机会执行一次 Minor GC.Minor Gc 通 常发生在新生代的 Eden 区,在这个区的对象生存期短,每每发生 Gc 的频率较高,回收速度比较快;Full Gc/Major GC 发生在老年代,通常状况下,触发老年代 GC的时候不会触发 Minor GC,可是经过配置,能够在 Full GC 以前进行一次 Minor GC 这样能够加快老年代的回收速度。

Java 类加载过程?

Java 类加载须要经历一下 7 个过程:

1. 加载

加载是类加载的第一个过程,在这个阶段,将完成一下三件事情:

• 经过一个类的全限定名获取该类的二进制流。

• 将该二进制流中的静态存储结构转化为方法去运行时数据结

构。

• 在内存中生成该类的 Class 对象,做为该类的数据访问入口。

2. 验证

验证的目的是为了确保 Class 文件的字节流中的信息不会危害到虚拟机.在该阶段主要完成如下四种验证: • 文件格式验证:验证字节流是否符合 Class 文件的规范,如主次版本号是否在当前虚拟机范围内,常量池中的常量是否有不被支持的类型. • 元数据验证:对字节码描述的信息进行语义分析,如这个类是否有父类,是否集成了不被继承的类等。

• 字节码验证:是整个验证过程当中最复杂的一个阶段,经过验证数据流和控制流的分析,肯定程序语义是否正确,主要针对方法体的验证。如:方法中的类型转换是否正确,跳转指令是否正确等。

• 符号引用验证:这个动做在后面的解析过程当中发生,主要是为了确保解析动做能正确执行。

3. 准备

准备阶段是为类的静态变量分配内存并将其初始化为默认值,这些内存都将在方法区中进行分配。准备阶段不分配类中的实例变量的内存,实例变量将会在对象实例化时随着对象一块儿分配在 Java 堆中。

public static int value=123;//在准备阶段 value 初始值为 0 。在初始化阶段才会变为 123 。

4. 解析

该阶段主要完成符号引用到直接引用的转换动做。解析动做并不必定在初始化动做完成以前,也有可能在初始化以后。

5. 初始化

初始化时类加载的最后一步,前面的类加载过程,除了在加载阶段用户应用程序能够经过自定义类加载器参与以外,其他动做彻底由虚拟机主导和控制。到了初始化阶段,才真正开始执行类中定义的Java 程序代码。

6. 使用

7. 卸载

描述一下 JVM 加载 Class 文件的原理机制**?**

Java 语言是一种具备动态性的解释型语言,类(Class)只有被加载到 JVM 后才能运行。当运行指定程序时,JVM 会将编译生成

的 .class 文件按照需求和必定的规则加载到内存中,并组织成为一个完整的 Java 应用程序。这个加载过程是由类加载器完成,具

体来讲,就是由 ClassLoader 和它的子类来实现的。类加载器自己也是一个类,其实质是把类文件从硬盘读取到内存中。

类的加载方式分为隐式加载和显示加载。

隐式加载指的是程序在使用 new 等方式建立对象时,会隐式地调用类的加载器把对应的类加载到 JVM 中。显示加载指的是经过直接调用 class.forName() 方法来把所需的类加载到 JVM 中。任何一个工程项目都是由许多类组成的,当程序启动时,只把须要的类加载到 JVM 中,其余类只有被使用到的时候才会被加载,采用这种方法一方面能够加快加载速度,另外一方面能够节约程序运行时对内存的开销。此外,在 Java 语言中,每一个类或接口都对应一个 .class 文件,这些文件能够被当作是一个个能够被动态加载的单元,所以当只有部分类被修改时,只须要从新编译变化的类便可,而不须要从新编译全部文件,所以加快了编译速度。在 Java 语言中,类的加载是动态的,它并不会一次性将全部类所有加载后再运行,而是保证程序运行的基础类(例如基类)彻底加载到 JVM 中,至于其余类,则在须要的时候才加载。

类加载的主要步骤:

• 装载。根据查找路径找到相应的 class 文件,而后导入。

• 连接。连接又可分为 3 个小步:

• 检查,检查待加载的 class 文件的正确性。

• 准备,给类中的静态变量分配存储空间。

• 解析,将符号引用转换为直接引用(这一步可选)

• 初始化。对静态变量和静态代码块执行初始化工做。