作者:gnuhpc
出处:http://www.cnblogs.com/gnuhpc/
1.类的加载
加载:查找并加载类的二进制数据。把.class文件的二进制数据读入到内存中,把它存放在运行时数据区的方法区内,然后在堆区中创建一个java.lang.Class对象(这就是类加载的最终产品),用来封装类在方法区内的数据结构,并且向JAVA程序提供了访问类在方法区内的数据结构的接口。
2.类的验证:
验证:确保加载类的正确性,检查的内容包括,类文件的结构检查,语义检查,字节码验证,二进制兼容的验证。
3.类的准备:
为类的静态变量分配内存,并将其初始化为默认值)和解析类的二进制数据(把类中的符号转换为直接引用)。
4.类的解析:
将类中的引用替换为指针。
5.类的初始化:
初始化:给类的静态变量赋予正确的初始值,JVM只有在程序首次主动使用一个类或者接口的时候才会初始化它(所谓主动使用,创建类的实例,调用类的静态方法,调用反射方法,初始化一个类的子类等都属于这一类)。这个类还没有被加载或连接,则先进行加载和连接,若类存在直接父类,则先初始化直接父类,接着在执行初始化语句,然后这个类的初始化才算完成。
以下是几种不会导致类的初始化的类被动使用的情况:
a.对于final类型的静态变量,如果在编译的时候就可以算出变量的取值,那么就不会导致类的初始化,而是直接在字节流中保存那个值,所以当访问的时候其实就是访问的那个值。例如:
public static final int a = 2*3;则public static final int b = (int) (Math.random()*5)就会导致类的初始化。
b.初始化一个类的时候,要求它的所有父类都已经被初始化了,但是这条规则并不适用于接口。但在初始化一个类时,并不会先初始化它所实现的接口。在初始化一个接口时,并不会先初始化它的父接口。
c.调用ClassLoader类的loadClass()方法加载一个类,并不是对类的主动使用,不会导致类的初始化。
类的生命周期:
当Sample类被加载、连接和初始化后,它的生命周期就开始了。当代表Sample类的Class对象不再被引用,即不可触及时,Class对象就会结束生命周期,Sample类在方法区内的数据也会被卸载,从而结束Sample类的生命周期。
JVM自身的类加载器所加载的类,在JVM生命周期中始终不会被卸载。而用户自定义的类加载器所加载的类是可以被卸载的。