深入理解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中的内存满了的情况下
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值