龙空技术网

【教3妹学编程-java】详解父子类变量、代码块、构造函数执行顺序

程序员小2 100

前言:

现时咱们对“java 父类和子类构造方法”可能比较关注,看官们都需要了解一些“java 父类和子类构造方法”的相关资讯。那么小编同时在网络上搜集了一些对于“java 父类和子类构造方法””的相关资讯,希望姐妹们能喜欢,小伙伴们快快来学习一下吧!

#记录我的2023#

-----------------第二天------------------------

本文先论述父子类变量、代码块、构造函数执行顺序的结论, 然后通过举例论证,接着再扩展,彻底搞懂静态代码块、动态代码块、构造函数、父子类、类加载机制等知识体系。温故而知新,建议点赞收藏~

1 先说结论

面试官:好的,你说一下java中父类子类的变量、代码块、构造函数执行顺序是怎样的?

3妹:好的,

Java程序中类中个元素的初始化顺序 初始化的原则是

先初始化静态部分,再初始化动态部分,先初始化父类部分,后初始化子类部分,先初始化变量,再初始化代码块和构造器。

具体的,由于类中初始化的部分有静态成员变量、静态代码块、普通成员变量、动态代码块、构造函数。所以跟父类子类组合起来有10种情况,总体顺序是:

1.父类的静态成员变量(如果是第一次加载类)2.父类的静态代码块(如果是第一次加载类)3.子类的静态成员变量(如果是第一次加载类)4.子类的静态代码块(如果是第一次加载类)5.父类的普通成员变量6.父类的动态代码块7.父类的构造器方法8.子类的普通成员变量9.子类的动态代码块10.子类的构造器方法2 举例

面试官:能写个栗子,验证一下吗

3妹:好的,假设有一个Parent和Child类

public class Parent {    static Instance staticInstance = new Instance("1---Parent类的静态成员变量staticInstance");    static  {        System.out.println("2---Parent类的静态代码块执行了");    }    Instance instance = new Instance("5---Parent类的普通成员变量instance");    {        System.out.println("6---Parent类的动态代码块执行了");    }    Base() {        System.out.println("7---Parent类的构造器方法执行了");    }}public class Child extends Parent {    static Instance staticInstance = new Instance("3---Child类的静态成员变量staticInstance");    static  {        System.out.println("4---Child类的静态代码块执行了");    }    Instance instance = new Instance("8---Child类的普通成员变量instance");    {        System.out.println("9----Child类的动态代码块执行了");    }    Child() {        System.out.println("10---Child类的构造器方法执行了");    }  //执行测试    public static void main(String[] args) {        Child child = new Child();    }}

测试结果如下:

1---Parent类的静态成员变量staticInstance进行了初始化

2---Parent类的静态代码块执行了

3---Child类的静态成员变量staticInstance进行了初始化

4---Child类的静态代码块执行了

5---Parent类的普通成员变量instance进行了初始化

6---Parent类的动态代码块执行了

7---Parent类的构造器方法执行了

8---Child类的普通成员变量instance进行了初始化

9----Child类的动态代码块执行了

10---Child类的构造器方法执行了

说明确实是按照上面的执行顺序执行的。

3 静态变量、静态代码块

面试官:好的,我们先来说一个静态变量和静态代码块,他们的加载时机是怎样的

3妹

静态变量、静态代码块是被static修饰的,是属于当前类的信息,类加载过程是先将编译后的class文件加载到内存中,一个类只会被加载到内存中一次。而static修饰的代码块属于类的信息的,所以,静态代码块中的代码有且只有一次被执行。执行的时机:类被加载的时候

4 动态代码块

面试官:那动态代码块呢?

3妹

动态代码块即不是static修饰的代码块,是用来初始化类实例信息的。当我们new关键字创建一个对象的时候,就会被执行,而且每使用一个new关键字创建出一个新对象的时候就会被执行一次的,在构造函数主题代码执行之前被运行的。

5 父类与子类的构造函数

面试官:再来说一下构造函数,new 一个子类的时候,父类的构造函数是何时被加载的呢?

3妹

父类与子类的加载时机:父类在子类前面

需要注意的是:子类的构造方法,不管是无参构造还是有参构造,默认都会先去寻找父类的无参构造方法。如果父类中,没有无参构造,那么子类必须使用supper这个关键字来调用父类带参数的构造方法,否则在编译期都不能通过。

如下图:

6 类的加载及初始化

面试官:再来说一下类的加载及初始化是怎样的?

3妹:好的:

Java类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using) 和 卸载(Unloading)七个阶段。其中准备、验证、解析3个部分统称为连接(Linking)。

类加载的时机:

什么情况下虚拟机需要开始加载一个类呢?虚拟机规范中并没有对此进行强制约束,这点可以交给虚拟机的具体实现来自由把握。

类初始化则是类加载的最后一步,

类初始化的时机

虚拟机规范指明 有且只有 五种情况必须立即对类进行初始化(而这一过程发生在加载、验证、准备之后):

1、遇到new、getstatic、putstatic或invokestatic这四条字节码指令(注意,newarray指令触发的只是数组类型本身的初始化,而不会导致其相关类型的初始化,比如,new String[]只会直接触发String[]类的初始化,也就是触发对类[Ljava.lang.String的初始化,而直接不会触发String类的初始化)时,如果类没有进行过初始化,则需要先对其进行初始化。生成这四条指令的最常见的Java代码场景是:

1)使用new关键字实例化对象的时候;

2) 读取或设置一个类的静态字段(被final修饰,已在编译器把结果放入常量池的静态字段除外)的时候;

3)调用一个类的静态方法的时候。2、使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。3、当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。4、当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。5、当使用jdk1.7动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getstatic,REF_putstatic,REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行初始化,则需要先出触发其初始化。7 再次总结一下

面试官:好的, 今天的面试就到这里,很高兴你能来参加面试~

3妹:谢谢面试官,再见~

3妹:今天的面试面试的还挺细的,我要总结一下:

Java程序中类中个元素的初始化顺序 初始化的原则是

先初始化静态部分,再初始化动态部分,先初始化父类部分,后初始化子类部分,先初始化变量,再初始化代码块和构造器。

具体的,由于类中初始化的部分有静态成员变量、静态代码块、普通成员变量、动态代码块、构造函数。所以跟父类子类组合起来有10种情况,总体顺序是:

1.父类的静态成员变量(如果是第一次加载类)2.父类的静态代码块(如果是第一次加载类)3.子类的静态成员变量(如果是第一次加载类)4.子类的静态代码块(如果是第一次加载类)5.父类的普通成员变量6.父类的动态代码块7.父类的构造器方法8.子类的普通成员变量9.子类的动态代码块10.子类的构造器方法

还有静态代码块、动态代码块、构造函数、父子类、类加载、类初始化,这些知识点虽然普通,但是也要牢记哦~

标签: #java 父类和子类构造方法