JVM学习笔记8——接口初始化规则与类加载器准备阶段和初始化阶段的重要意义分析

本文探讨了Java中接口继承时如何处理常量初始化,即使删除父接口的实现类,子接口的常量仍能保持有效。重点讲解了接口变量默认为final的影响,以及实例化时机对常量值的影响,通过实例演示了不同情况下的行为差异。
摘要由CSDN通过智能技术生成
public class MyTest5 {

    public static void main(String[] args){

        System.out.println(MyChild5.b);

    }

}

interface MyParent5 {
    public static final int a = 5;
}

interface MyChild5 extends MyParent5 {
    public static final int b = 6;
}

程序会输出6,即使删掉MyParent5的class文件程序也能正常运行,(接口里的变量默认都是final的)说明

当一个接口在初始化时,并不要求其父接口都完成了初始化

MyChild5中的b改为

// 如果删掉MyParent5的class文件会报错
public static final int b = new Random().nextInt(2); 
public class MyTest5 {

    public static void main(String[] args){

        System.out.println(MyChild5.b);
    }

}

interface MyParent5 {
    public static final int a = new Random().nextInt(3); 
}

interface MyChild5 extends MyParent5 {
    public static final int b = 5;
}

删掉MyParent5是可以正常执行

public class MyTest5 {

    public static void main(String[] args){

        System.out.println(MyChild5.b);
    }

}

interface MyParent5 {
    public static final int a = new Random().nextInt(3); 
}

interface MyChild5 extends MyParent5 {
    public static final int b = new Random().nextInt(4);
}

删掉MyParent5会报错

只要在真正使用到父接口的时候(如引用接口中定义的常量时),才会初始化

public class MyTest6 {

    public static void main(String[] args){
        Singleton singleton = Singleton.getInstance();

        System.out.println("counter1: " + Singleton.counter1);
        System.out.println("counter2: " + Singleton.counter2);
    }

}

class Singleton{
    public static int counter1;
    public static int counter2 = 0;

    public static Singleton singleton = new Singleton();

    private Singleton(){
        counter1 ++;
        counter2 ++;
    }

    public static Singleton getInstance(){
        return singleton;
    }
}

 执行结果如下:

 将Singleton的counter2放在私有构造方法之后

class Singleton{

    public static int counter1;

    public static Singleton singleton = new Singleton();

    private Singleton(){
        counter1 ++;
        counter2 ++; // 主动使用,在准备阶段赋上初值
        // System.out.println("counter1: " + counter1);
        // System.out.println("counter2: " + counter2);
    }

    // 代码移到这里输出就变成了0,执行到这里时相当于又手工赋值0
    public static int counter2 = 0; 

    public static Singleton getInstance(){
        return singleton;
    }
}

执行结果如下:

初始化的顺序是在代码里声明的位置从上到下,所以counter2在准备阶段是赋值为默认值0,然后在构造方法里加1,但是之后初始化又被重新赋值为指定的0

在私有构造方法中打印出来:

public class MyTest6 {

    public static void main(String[] args){
        Singleton singleton = Singleton.getInstance();

        System.out.println("counter1: " + Singleton.counter1);
        System.out.println("counter2: " + Singleton.counter2);
    }

}

class Singleton{
    public static int counter1;

    public static Singleton singleton = new Singleton();

    private Singleton(){
        counter1 ++;
        counter2 ++; // 主动使用,在准备阶段赋上初值,准备阶段的意义
        System.out.println("counter1: " + counter1);
        System.out.println("counter2: " + counter2);
    }

    // 代码移到这里输出就变成了0,执行到这里时相当于又手工赋值0
    public static int counter2 = 0;

    public static Singleton getInstance(){
        return singleton;
    }
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值