如果类没有被加载到内存中,则先进行类加载
类中的静态代码块和静态属性初始化值(按代码顺序执行)
创建对象时发生了什么?
非静态代码块和非静态属性初始化值(按代码顺序执行),然后调用构造方法
下面这个例子我们创建B的一个对象,B类之前没被加载,B类继承了A。A中有两个属性,一个静态的staticNum和一个非静态的num,分别用静态方法staticMethod和非静态方法method为其赋值。
package stringTest;
/**
* writer: holien
* Time: 2017-11-25 10:09
* Intent: 父类子类的加载顺序
*/
public class LoadingOrder {
public static void main(String[] args) throws Exception {
B b = new B();
// Class.forName("stringTest.B"); // 执行静态代码块和为静态属性初始化值
// ClassLoader.getSystemClassLoader().loadClass("stringTest.B"); // 不执行,只加载类
}
}
class A {
// 静态代码块和静态属性按顺序执行(类加载阶段)
static int staticNum = staticMethod();
static {
System.out.println("A静态代码块");
}
// 非静态属性初始化值和非静态代码块顺序执行(创建对象阶段)
int num = method();
{
System.out.println("A代码块");
}
// 最后才调用构造方法
A() {
System.out.println("A构造方法");
}
static int staticMethod() {
System.out.println("A调用静态方法为静态属性初始化值");
return 5;
}
int method() {
System.out.println("A调用普通方法为属性初始化值");
return 5;
}
}
class B extends A {
static int num2 = staticMethod1();
static {
System.out.println("B静态代码块");
}
static int staticMethod1() {
System.out.println("B调用静态方法为静态属性初始化值");
return 5;
}
{
System.out.println("B代码块");
}
B() {
System.out.println("B构造方法");
}
}
执行结果:
A调用静态方法为静态属性初始化值
A静态代码块
B调用静态方法为静态属性初始化值
B静态代码块
A调用普通方法为属性初始化值
A代码块
A构造方法
B代码块
B构造方法
可以看到,类加载阶段先父后子,创建对象也是先父后子。
Class.forName()与ClassLoader.getSystemClassLoader().loaderClass()这两种类加载方式有所区别,Class.forName()会执行静态代码块并且为静态属性初始化值,而classLoader只负责把类从硬盘或网络中加载到内存中,不进行连接,也就没有后面的初始化,所以不会执行。这也是为什么获取JDBC驱动时要用Class.forName()的原因,其源码是在静态代码块中获取驱动对象的,如下:
public class Driver extends NonRegisteringDriver implements java.sql.Driver {
static {
try {
//往DriverManager中注册驱动
java.sql.DriverManager.registerDriver(new Driver());
} catch (SQLException E) {
throw new RuntimeException("Can't register driver!");
}
}
public Driver() throws SQLException {
}
}