1、Java虚拟机与程序的生命周期。在如下几种情况下,Java虚拟机将结束生命周期
– 执行了System.exit()方法
– 程序正常执行结束
– 程序在执行过程中遇到了异常或错误而异常终止
– 由于操作系统出现错误而导致Java虚拟机进程终止
2、类的加载、连接和初始化
1) 加载:查找并加载类的二进制数据
2) 连接
– 验证:确保被加载的类的正确性
– 准备:为类的静态变量分配内存,并将其初始化为默认值 (只有静态变量)
– 解析:把类中的符号引用转换为直接引用
3) 初始化:为类的静态变量赋予正确的初始值
3) 初始化:为类的静态变量赋予正确的初始值
【如】:在连接的准备阶段为静态变量分配空间(int为4个字节,long为8个字节)并且初始化为默认值0。在初始化阶段变量被初始化为3。
1
2
3
|
public class Test {
private static int a = 3 ; // a=0 ——> a=3
}
|
或
1
2
3
4
5
6
7
|
public class Test {
private static int a; // a = 0
static {
a = 3 ; // a = 3
}
}
|
在准备阶段,Java虚拟机为类的静态变量分配内存,并设置默认的初始值。例如对于以下Sample类,在准备阶段,将为int类型的静态变量a分配4个字节的内存空间,并且赋予默认值0,为long类型的静态变量b分配8个字节的内存空间,并且赋予默认值0。过程:
3、所有的Java虚拟机实现必须在每个类或接口被Java程序“首次主动使用”时才初始化他们 。Java程序对类的使用方式可分为两种
1) 主动使用
– 创建类的实例 (如:new Object();)
– 访问某个类或接口的静态变量,或者对该静态变量赋值(如:int a = Test.COUNT; )
– 调用类的静态方法 (如:Test.getInstance();)
– 反射(如:Class.forName(“com.shengsiyuan.Test”))
– 初始化一个类的子类
– Java虚拟机启动时被标明为启动类的类(如: c:\java Test)
2) 被动使用
除了以上六种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化。
4、类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构
1) 加载.class文件的方式
– 从本地系统中直接加载
– 通过网络下载.class文件
– 从zip,jar等归档文件中加载.class文件
– 从专有数据库中提取.class文件
– 将Java源文件动态编译为.class文件
5、类的加载的最终产品是位于堆区中的Class对象。Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口
1)有两种类型的类加载器
– Java虚拟机自带的加载器
a)根类加载器(使用C++编写,程序员在java代码中无法获取源代码)(Bootstrap)
b)扩展类加载器,使用java代码实现(Extension)
c)系统类加载器(应用加载器),java代码实现(System)
– 用户自定义的类加载器
a)java.lang.ClassLoader的子类
b)用户可以定制类的加载方式
c)类加载器并不需要等到某个类被“首次主动使用”时再加载它
6、 JVM规范允许类加载器在预料某个类将要被使用时就预先加载它,如果在预先加载的过程中遇到了.class文件缺失或存在错误,类加载器必须在程序首次主动使用该类时才报告错误(LinkageError错误) 如果这个类一直没有被程序主动使用,那么类加载器就不会报告错误 。 类被加载后,就进入连接阶段。连接就是将已经读入到内存的类的二进制数据合并到虚拟机的运行时环境中去。