加载过程
JVM将类加载分为三个步骤
(1)装载:装载时查找对应的二进制数据
(2)链接:验证正确性,分配内存,将符号引用转化为引用
(3)初始化:为静态变量赋予正确的初始值。也就是staitic
类加载将类的.class文件中的二进制数据读入内存中。然后在对去创建这个类中的java.lang.class对象。
类加载器:
类加载子系统是由类加载器完成的
Bootstrap ClassLoader负责加载jre/lib/rt.jar的所有子类。是由C++实现的。
Extension ClassLoader 负责加载$JAVA_HOME下jre/lib/*.jar中的jar
App ClassLoader 负责加载classpath中指定的jar包及目录中class
Custom ClassLoader 根据自身需要定义的类加载器。如jboss或者是tomcat都会实现自己的类加载器。
除了Bootstrap ClassLoader,其他所有类加载器都是ClassLoader的子类。
加载之前,应用程序会自底向上检查类型是否已经被加载。如果确定没有被加载,则从上到下依次尝试加载这个类。
父类的类加载
public class Sandbox4 extends Sandbox2{
public static void main(String[] args) {
Test1 t1 = new Test1();
}
}
class Test1 extends Test2{
static{
System.out.println("static area in Test1 called");
}
Test1(){
System.out.println("constructor Test1 called");
}
}
class Test2
{
static{
System.out.println("static area in Test2 called");
}
Test2(){
System.out.println("constructor Test2 called");
}
}
输出为
static area in Test2 called
static area in Test1 called
constructor Test2 called
constructor Test1 called
也就是说,是先初始化父类的静态部分,再初始化子类的静态部分,在运行父类的构造函数,在运行子类的构造函数。(子类的构造函数第一句默认是父类的构造函数,如果你不写,系统自动就会把父类的无参构造函数放在这里)
在反射机制中的类加载
public class Sandbox4 extends Sandbox2{
public static void main(String[] args) {
try {
Class.forName("Pack2.Test1");
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("finished");
}
void ChangeString(String s){
s = "test2";
}
}
class Test1{
static{
System.out.println("static area in Test1 called");//会被输出
}
Test1(){
System.out.println("constructor Test1 called");//不会被输出
}
}
使用class.forname时,类加载器将类加载进内存,并不会创建对象。创建对象应该使用如下语句
Class classType = Class.forName("lxf.Person");
Object obj = classType.newInstance();