Java程序对类的使用方式可分为两种
- 主动使用
- 创建类的实例
- 访问某个类或接口的静态变量(getstatic)、或者对该静态变量赋值(putstatic)
- 调用类的静态方法(invokestatic)
- 反射(如Class.forName(“com.test.Test”))
- 初始化一个类的子类
- Java虚拟机启动时被标明为启动类的类(Java Test)
- JDK1.7开始提供动态语言支持:java.lang.invoke.MethodHandle实例的解析结果REF_getStatic,REF_putStatic,REF_invokeStatic句柄对应的类没有初始化,则初始化 - 被动使用
除了以上七种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化 - 所有Java虚拟机实现必须在每个类或接口被Java程序首次主动使用时才初始化他们
类的加载
类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后再内存中创建一个Java.lang.Class对象(规范并未说明Class对象位于哪里,HotSpot虚拟机将其放在了方法区中)用来封装类在方法区内的数据结构
加载.class文件的方式:
- 从本地系统中直接加载
- 通过网络下载.class文件
- 从zip,jar等归档文件中加载.class文件
- 从专有数据库中提取.class文件
- 将Java源文件动态编译为.class文件
访问某个类的静态变量会触发初始化,对于静态字段来说,只有直接定义了该字段的类才会被初始化
/*
* Copyright (c) 2019 maoyan.com
* All rights reserved.
*
*//**
* JVM 初始化
*
* -XX:+TraceClassLoading,用于追踪类的加载信息并打印出来
* -XX:+<option> 表示开启option选项
* -XX:-<option> 表示关闭option选项
* -XX:<option>=<value> 表示将option选项赋值为value
*
* @author wangkai
* @created 2019/12/27
*/publicclassMyTest{publicstaticvoidmain(String[] args){//对于静态字段来说,只有直接定义了该字段的类才会被初始化//类加载器并不需要等到某个类被首次使用
System.out.println(MyChild.str1);}}classMyParent{static String str1="wangkai";static{
System.out.println("my parent static block");}}classMyChildextendsMyParent{static String str2="wangkai1";static{
System.out.println("my child static block");}}
my parent static block
wangkai
Process finished withexit code 0
当一个子类被初始化的时候,要求其父类也完全初始化
/*
* Copyright (c) 2019 maoyan.com
* All rights reserved.
*
*//**
* JVM 初始化
*
* -XX:+TraceClassLoading,用于追踪类的加载信息并打印出来
* -XX:+<option> 表示开启option选项
* -XX:-<option> 表示关闭option选项
* -XX:<option>=<value> 表示将option选项赋值为value
*
* @author wangkai
* @created 2019/12/27
*/publicclassMyTest{publicstaticvoidmain(String[] args){//当一个类被初始化的时候,要求其父类已经完成初始化
System.out.println(MyChild.str2);}}classMyParent{static String str1="wangkai";static{
System.out.println("my parent static block");}}classMyChildextendsMyParent{static String str2="wangkai1";static{
System.out.println("my child static block");}}