深入理解JVM 一:类加载器
类加载
- 在Java代码中,类型的加载、连接与初始化过程都是在程序运行期间完成的
- 提供了更大的灵活性,增加了更多的可能性
类加载器
-
java虚拟机与程序的生命周期
-
在如下几种情况下,Java虚拟机将结束生命周期
- 执行了System.exit()方法
- 程序正常执行结束
- 程序在执行过程中遇到了异常或者错误而异常终止
- 由于操作系统出现错误而导致java虚拟机进程终止
类的加载、连接与初始化
-
加载:查找并加载类的二级制数据
-
连接
- -验证: 确保被加载的类的正确性
- -准备: 为类的静态变量分配内存,并将其初始化为默认值
- -解析: 把类中的符号引用转换为直接引用
-
初始化: 为类的静态变量赋予正确的初始值
-
Java程序的使用方式可分为两种
- -主动使用
- -被动使用
-
所有的Java虚拟机实现必须在每个类或接口被Java程序“首次主动使用”时才初始化他们
-
主动使用
- -创建类的实例
- -访问某个类或接口的静态变量,或对该静态变量赋值
- -调用类的静态方法
- -反射(如Class.forName(“com.test.Test”))
- -初始化一个类的子类
- -Java虚拟机启动时被标明为启动类的类
- JDK1.7开始提供的动态语言支持:java.lang.invoke.MethodHandle实例的解析结果REF_getStatic,REF_putStatic,REF_invokeStatic句柄对应的类没有初始化则初始化
-
除了上述的方式外,其它使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化
示例:
/*
对于静态代码字段来说,只有定义了该字段的类才会被初始化;
当一个类在初始化时,要求其父类全部都已经初始化完毕了
-XX:+TraceClassLoading,用于追踪类加载信息并打印出来
-XX:+<option> ,表示开启option选项
-XX:-<option> ,表示关闭option选项
-XX:<option>=<value> ,表示将option选项的值设置为value
*/
public class MyTest(){
public static void main(String args){
System.out,prinln(MyChild1.str);
}
}
class MyParent1{
public static String str="hello world";
static {
System.out,prinln("MyParent static block");
}
}
class MyChild1 extends MyParent1{
public static String str2="welcome";
static {
System.out,prinln("MyChild static block");
}
}
打印结果是:
MyParent static block
hello world
我们把代码改成调用MyChild1中的str2
public static void main(String args){
System.out,prinln(MyChild1.str2);
}
打印结果是:
MyParent static block
MyChild static block
welcome
由此我们可以知道,对子类静态方法的访问是主动使用,静态变量是谁的就初始化该类。Mychild1.str的使用是调用父类的属性,所以是对父类的初始化。而Mychild1.str2是使用子类的属性,但是当一个类在初始化时,要求其父类全部都已经初始化完毕了,所以父类也被初始化而调用父类的静态代码块。
再把代码改成(加多final关键字):
public class MyTest(){
public static void main(String args){
System.out,prinln(MyChild1.str);
}
}
class MyParent1{
public static final String str="hello world";
static {
System.out,prinln("MyParent static block");
}
}
结果:
hello world
常量在编译阶段会存入到调用这个常量方法所在的类的常量池中,本质上,调用类并没有直接引用到定义常量的类,因此并不会触发定义常量的类的初始化。静态代码块就不会执行。
我们再看这段代码,静态代码块是否会执行?
public class MyTest(){
public static void main(String args){
MyParent1[] myParents=new MyParent1[1];
}
}
class MyParent1{
static {
System.out,prinln("MyParent static block");
}
}
没有任何输出
说明对于数组实例,其类型是由JVM在运行期动态生成的,表示为[包.Myparent这种形式。动态生成的类型,其父类型就是Object.对数组来说,javaDoc经常将构成数组元素为Component,实际上就是将数组降低后的一个类型。
类的初始化时机
当java虚拟机初始化一个类时,要求它的所有父类都已经被初始化,但是这条规则并不适用于接口。
- 在初始化一个类时,并不会先初始化它所实现的接口。
- 在初始化一个接口时,并不会先初始化它的父接口。
因此,一个父接口并不会因为它的子接口或者实现类的初始化而初始化,只有当程序首次使用特定接口的静态变量时,才会导致该接口的初始化。
GC
虚拟机中非常低的一个守护线程
- GC执行时机:
- 对象回收会调用finaize(),意味着这个对象就是从内存中销毁掉了吗?
finiaize()提醒GC该关心我了,GC强引用指向该对象,强引用:A a=new A();GC不会主动回收,宁愿出现OOM错误。 - a=null ,执行GC回收的时候,当内存满了,GC才会执行回收操作。
条件:- 1、堆内存中对象没有任何引用指向
- 2、 jvm中的内存满了的情况下
- 对象回收会调用finaize(),意味着这个对象就是从内存中销毁掉了吗?