创建对象时JVM为我们做了什么

如果类没有被加载到内存中,则先进行类加载

类中的静态代码块和静态属性初始化值(按代码顺序执行)

创建对象时发生了什么?

非静态代码块和非静态属性初始化值(按代码顺序执行),然后调用构造方法


下面这个例子我们创建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 {
    }
}

你可能感兴趣的:(java)