【Java笔记】java中静态代码块、构造块、构造方法的执行时机

本文详细探讨了Java中的静态代码块、构造块和构造方法的执行时机。静态代码块用于初始化static属性,只执行一次;构造块在每次创建对象时调用,优先于构造方法;子类对象创建时,先执行父类的初始化。文章通过多个示例分析了不同场景下的执行顺序,并列举了类被主动使用时的多种情况。

总结:

静态块的主要作用是为static属性进行初始化, 无论产生多少实例化对象,静态块都只执行一次。

构造块(非静态代码块)每产生一个新的对象就调用一次,且优先于构造方法执行。

产生子类对象时,由于子类继承了父类,所以父类先执行。

下面写几个范例:

Java代码:

1.有继承关系,子类是主类。

class HelloA {
    //构造方法
    public HelloA(){
        System.out.println("Hello A!父类构造方法");
    }
    //非静态代码块
    {
        System.out.println("i'm A class.父类非静态代码块");
    }
    //静态代码块
    static{
        System.out.println("static A 父类静态代码块");
    }
}

public class HelloB extends HelloA {
    //构造方法
    public HelloB(){
        System.out.println("Hello B! 构造方法");
    }
    //非静态代码块
    {
        System.out.println("i'm B class.非静态代码块");
    }
    //静态代码块
    static{
        System.out.println("static B 静态代码块");
    }
    public static void main(String[] args) {
        System.out.println("---start---");
        new HelloB();
        new HelloB();
        System.out.println("---end---");
    }
}

结果:

分析一:

1.静态代码块:

由于HelloB是主类,而main方法是程序的入口,此时HelloB这个类被主动使用,要对其进行初始化,而HelloB又继承了HelloA,所以HelloA先初始化,初始化时父类的静态代码块被执行,接下来HelloB初始化,子类的静态代码块被执行

2.构造块(也就是上述类中的非静态代码块)和构造方法:

进入主函数后,依次创建了两次HelloB的匿名对象。

构造块和构造方法都是在创建对象时调用,构造块优先于构造方法执行。

HelloB是HelloA的子类,HelloA的构造块和构造方法先执行,再执行HelloB的构造块和构造方法

这里也说明了无论产生多少实例化对象,静态块都只执行一次。

 

2.有继承关系,父类、子类都不是主类

class HelloA {
    //构造方法
    public HelloA(){
        System.out.println("Hello A!父类构造方法");
    }
    //非静态代码块
    {
        System.out.println("i'm A class.父类非静态代码块");
    }
    //静态代码块
    static{
        System.out.println("static A 父类静态代码块");
    }
}

class HelloB extends HelloA {
    //构造方法
    public HelloB(){
        System.out.println("Hello B! 构造方法");
    }
    //非静态代码块
    {
        System.out.println("i'm B class.非静态代码块");
    }
    //静态代码块
    static{
        System.out.println("static B 静态代码块");
    }
}
public class TestHelloB {
    static{
        System.out.println("TestHelloB 静态代码块");
    }
    public static void main(String[] args) {
        System.out.println("---start---");
        new HelloB();
        System.out.println("---end---");
    }
}

结果:

3.无继承关系,父类、子类都不是主类

class HelloA {
    //构造方法
    public HelloA(){
        System.out.println("Hello A构造方法");
    }
    //非静态代码块
    {
        System.out.println("i'm A class.非静态代码块");
    }
    //静态代码块
    static{
        System.out.println("static A 静态代码块");
    }
}

class HelloB {
    //构造方法
    public HelloB(){
        System.out.println("Hello B 构造方法");
    }
    //非静态代码块
    {
        System.out.println("i'm B class.非静态代码块");
    }
    //静态代码块
    static{
        System.out.println("static B 静态代码块");
    }
}
public class TestHelloB {
    public static void main(String[] args) {
        System.out.println("---start---");
        new HelloB();
        System.out.println("---end---");
    }
}

结果:

分析二:

对比代码2和代码3是为了再次强调,静态代码块在初始化执行,主类肯定是首先被主动使用的类,此时JVM就会对其初始化,

若主类是某一个类的子类,当然是先对它的父类进行初始化,初始化时父类有静态代码块,就会被优先执行,接下来才会执行子类的静态代码块。

若主类没有继承关系,初始化的当然就只有它自己了,此时若主类有静态代码块,此代码块就会被执行。

进入main函数后同理,创建对象时JVM就会对这个类初始化,若此类有父类,先对父类初始化。fe

分析三:

当一个类被主动使用时,JVM会对其初始化,整理主动使用的情况如下:

(1)当创建某个类的新实例时(如通过new或者反射,克隆,反序列化等)

(2)当调用某个类的静态方法时

(3)当使用某个类或接口的静态字段时

(4)当调用Java API中的某些反射方法时,比如类Class中的方法,或者java.lang.reflect中的类的方法时

(5)当初始化某个子类时

(6)当虚拟机启动某个被标明为启动类的类(即包含main方法的那个类)

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值