java 子类父类静态变量、方法、代码块、构造函数执行顺序探讨

重温类加载的过程:加载 验证 准备(为static类变量分配内存,初始化设置为null or 0) 解析 初始化(类变量赋值阶段)

 

1、虚拟机是优先加载含有main函数的类,会对此类优先加载,故要先加载类,才能执行main函数。

注:注意类加载时机,调用类的常量不加载。

2、加载类时候,若牵扯到父类子类,则一般要先调用父类,如果从子类调用父类的静态变量,不需要加载子类,只需要加载父类。

若两类都需要加载,则顺序是父类显式静态初始化成员变量、代码块 ------->子类显式静态初始化代码块 、成员变量

3、若实例化了此类,就会在静态参数加载完之后(仅一次),再去加载这个类的成员变量、非静态代码块,构造函数。

“编译器会将实例变量初始化和实例代码块初始化相关代码放到类的构造函数中去,并且这些代码会被放在对超类构造函数的调用语句之后,构造函数本身的代码之前。”

顺序是:父类非静态实例初始化代码块、成员变量(按顺序) > 父类构造函数 > 子类非静态实例初始化代码块 (按顺序)> 子类构造函数

 

如果要彻底的弄懂这个,就要知道类加载和类实例化的过程,这里不再赘述,仅作为笔记。

 

参考:

       我竟然不再抗拒 Java 的类加载机制了:https://zhuanlan.zhihu.com/p/73078336

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是Test.java文件的代码: ``` public class Test { public static void main(String[] args) { Leaf leaf1 = new Leaf(); Leaf leaf2 = new Leaf(); } } class Root { static { System.out.println("Root的静态代码块"); } { System.out.println("Root的实例代码块"); } public Root() { System.out.println("Root的构造函数"); } } class Mid extends Root { static { System.out.println("Mid的静态代码块"); } { System.out.println("Mid的实例代码块"); } public Mid() { this("调用重载的构造器"); System.out.println("Mid的无参构造函数"); } public Mid(String s) { System.out.println("Mid的带参构造函数:" + s); } } class Leaf extends Mid { static { System.out.println("Leaf的静态代码块"); } { System.out.println("Leaf的实例代码块"); } public Leaf() { super("调用父类指定的构造器"); System.out.println("Leaf的无参构造函数"); } } ``` 输出结果为: ``` Root的静态代码块 Mid的静态代码块 Leaf的静态代码块 Root的实例代码块 Root的构造函数 Mid的实例代码块 Mid的带参构造函数:调用重载的构造器 Mid的无参构造函数 Leaf的实例代码块 Leaf的带参构造函数:调用父类指定的构造器 Leaf的无参构造函数 Root的实例代码块 Root的构造函数 Mid的实例代码块 Mid的带参构造函数:调用重载的构造器 Mid的无参构造函数 Leaf的实例代码块 Leaf的带参构造函数:调用父类指定的构造器 Leaf的无参构造函数 ``` 可以看到,初始化代码块构造函数执行顺序符合题目中所给的加载顺序。同时,Leaf类使用super显示调用其父类指定的构造器,而Mid类使用this调用重载的构造器,验证了题目中的要求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值