JVM上分笔记 - 类的加载、连接与初始化(二)

再看本篇文章之前,请确保已经看过JVM- 类的加载、连接与初始化(一)

首先我们来看接口的初始化

public class Test5 {
    public static void main(String[] args) {
        System.out.println(Child5.a);
    }
}

interface Parent5 {
    public static final int a = 0;
}

interface Child5 extends Parent5 {
    public static final int b = 0;
}

接口无法像类一样,可以定义静态代码块来查看初始化的时机,所以在这里只能通过删除class文件的方式来观察,在删除class文件之后,如果没有报错,便证明接口没有初始化,如果报错,则接口初始化。

大家都知道在java中,接口中定义变量都是public static final的,这里只是显示的写出,通过删除编译的class文件,可以得出结论,当一个接口初始化时,并不要求其父接口都完成初始化,只有真正使用到父接口时,才初始化,这里一定要和类的初始化区分开,接口的初始化和类的初始化机制并不相同。

接下来在看一个例子,这个例子的代码在实际开发中肯定不会写出,但是对于我们理解类的连接和初始化大有帮助。

还是和上篇文章一样,先贴代码,小伙伴们先思考程序的输出,心里有了答案在往下看。

public class Test6 {
    public static void main(String[] args) {
        SingleTon singleTon = SingleTon.getInstance();
        System.out.println(SingleTon.a);
        System.out.println(SingleTon.b);
    }
}

class SingleTon {
    public static int a;
    public static int b = 0;

    private static SingleTon singleTon = new SingleTon();

    private SingleTon() {
        a++;
        b++;
    }

    public static SingleTon getInstance() {
        return singleTon;
    }
}

 

 

 

 

---------我是占位图----------

 

 

程序输出的是2 , 1

相信小伙伴们都答对了,接下来,改变一行代码的位置,将b的声明放在私有构造方法的下面,然后输出是什么呢

public class Test6 {
    public static void main(String[] args) {
        SingleTon singleTon = SingleTon.getInstance();
        System.out.println(SingleTon.a);
        System.out.println(SingleTon.b);
    }
}

class SingleTon {
    public static int a;

    private static SingleTon singleTon = new SingleTon();

    private SingleTon() {
        a++;
        b++;
    }

    public static int b = 0;

    public static SingleTon getInstance() {
        return singleTon;
    }
}

 

 

 

---------我是占位图----------

 

 

 

程序输出是1,0

为什么b的输出是0呢,回想上篇博客在介绍类连接的时候,有一个准备流程,会为静态变量申请内存同时赋予初值,此时,也就是准备阶段,a的初值为0,b的初值为0,singleton的初值为null。接下来在main函数里调用了getInstance()方法,这里属于主动调用,所以类要开始初始化,我们一行一行的来看

public static int a;这里没有显示的赋初值,所以在初始化的阶段,到这里a还是0。

private static SingleTon singleTon = new SingleTon(); 此时调用私有构造方法,a++, b++,此时a的值为1,b的值为1。

public static int b = 0;这里又显示的为b赋予了初值0,所以在初始化完成之后,b的值为0;

小伙伴们可以在私有的构造方法里加入打印语句,来验证上面的分析,看看在构造方法里完成++操作后,a和b的值是否都为1。

然后在改一下代码,原来a没有显示的赋初值,现在将a的值显示的声明为1

class SingleTon {
    public static int a = 1;

    private static SingleTon singleTon = new SingleTon();

    private SingleTon() {
        a++;
        b++;
    }

    public static int b = 0;

    public static SingleTon getInstance() {
        return singleTon;
    }
}

 

 

 

 

---------我是占位图----------

 

 

 

程序输出是2,0

相信都和小伙伴们想的一样,分析方法和上面一致,这里就不在赘述了,通过这个例子,相信小伙伴们对于类连接的准备过程和初始化过程,会有更进一步的认识。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值