类的加载、连接与初始化过程详解
类的加载、连接与初始化
1、过程:加载-->连接(验证-->准备-->解析) -->初始化
2、过程介绍:
A、加载:指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在内存中创建一个java.lang.Class对象(规范并未说明Class对象位于那里,HotSpot虚拟机将其放在了方法区中)用来封装类在方法区内的数据结构。反射---Class对象好像一面镜子来反射类在内存中的数据结构情况,可以利用Class对象将类的信息反应出来。
B、连接:
验证:进行字节码的校验和验证工作
准备:为静态变量开辟一块内存区域,并为其赋予对应类型的默认值
解析:将类之间的符号引用,变成直接引用
C、初始化:为静态变量赋予真正的值
类的加载的来源
1、从本地系统中直接加载
2、通过网络下载.class文件
3、从zip、jar等归档文件中加载.class文件
4、从专有的数据库中提取.class文件
5、将Java动态编译为.class文件(类似的还有动态代理,运行期创建类;jsp转换成servlet这个是一个动态创建class文件的过程)
Java程序对类的使用方式
可分为两种:
A、主动使用
B、被动使用
注:所有的java虚拟机实现必须在每个类或接口被java程序“首次主动调用”时才初始化它们。
主动使用(七种,重要)
1、创建类的实例。例:newObject()
2、访问某个类或接口的静态变量,或者是针对静态变量赋值。(调用使用getstatic指令获取,使用putstatic指令赋值)
3、调用类的静态方法。(invokestatic助记符)
4、反射(如Class.forName(“com.test.Test”);)
5、初始化一个类的子类
6、Java虚拟机启动时被标明为启动类的类(Java Test)
7、JDK1.7开始提供的动态语言支持:java.lang.invoke.Methodhandle实例解析结果REF_getstatic、REF_putstatic、REF_invokestatic句柄对应的类没有初始化,则进行初始化操作。
被动使用
除了以上七种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化。这里不会导致初始化,并不意味着不被加载,也有可能不加载。
例1:
public class MyTest1 {
publicstatic void main() {
System.out.println(MyChild1.str);
}
}
class MyParent1 {
publicstatic String str = “hello world”;
static{
System.out.println(“MyParent1static block”);
}
}
class MyChild1 extends MyParent1 {
static {
System.out.println(“MyChild1 static block”);
}
}
运行结果:
MyParent1 static block
helloworld
分析原因:
这里是对MyParent1的str主动使用,所以针对MyParent1进行初始化;虽然这里显示是MyChild1调用了str,但是却没有使用MyChild1。
例2:
public class MyTest1 {
publicstatic void main() {
System.out.println(MyChild1.str2);
}
}
class MyParent1 {
publicstatic String str = “hello world”;
static {
System.out.println(“MyParent1 static block”);
}
}
class MyChild1 extends MyParent1 {
publicstatic String str2 = “welcome”;
static {
System.out.println(“MyChild1 static block”);
}
}
运行结果:
MyParent1 static block
MyChild1 static block
welcome
分析原因:
这里针对MyChild1的主动使用,所以MyChild1会被初始化,其父类也会被初始化。
案例总结笔记:
1、对于静态字段来说,只有定义了该字段的类才会被初始化
2、当一个类初始化时,要求其父类全部都初始化完毕了(直到Object类也被初始化,并且每个类只被初始化一次)。