java中关于子类和父类的构造函数,静态代码块,普通代码块,静态成员变量和普通成员变量的初始化顺序

java中关于子类和父类的构造函数,静态代码块,普通代码块,静态成员变量和普通成员变量的初始化顺序

//父类:

class Parent{
    public static String i="我是父类的静态成员变量";
    public String j="我是父类普通的成员变量";

    public Parent() {
        System.out.println("我是父类构造");
    }
    static {
        System.out.println("我是父类静态代码块");
    }
    {
        System.out.println("我是父类代码块");
    }
}

//子类

public class Son extends Parent{
    public static String i="我是子类的静态成员变量";
    public String j="我是子类普通的成员变量";
    public Son () {
        System.out.println("我是子类构造");
    }
    static {
        System.out.println("我是子类静态代码块");
    }
    {
        System.out.println("我是子类代码块");
    }

}

//测试1

    public static void main(String[] args) {
        System.out.println(Son.i);
    }

输出结果:(只初始化静态代码块和静态成员变量,静态代码块和静态成员变量的初始化顺序有他们的编写顺序决定,并且父类先于子类)
在这里插入图片描述

//测试2

    public static void main(String[] args) {
        System.out.println(new Son().j);
    }

在这里插入图片描述

//测试3

public static void main(String[] args) {
       System.out.println(new Parent());
    }

在这里插入图片描述
//测试4

public static void main(String[] args) {
       System.out.println(new Son());
    }

在这里插入图片描述
//测试5

public static void main(String[] args) {
       System.out.println(new Son());
       System.out.println("------------------------");
        new Son();
    }

在这里插入图片描述

总结:

如果类还没有被加载:
1、先执行父类的静态代码块和静态变量初始化,并且静态代码块和静态变量的执行顺序只跟代码中出现的顺序有关。
2、执行子类的静态代码块和静态变量初始化。
3、执行父类的实例变量初始化
4、执行父类的构造函数
5、执行子类的实例变量初始化
6、执行子类的构造函数

如果类已经被加载:
则静态代码块和静态变量就不用重复执行,再创建类对象时,只执行与实例相关的变量初始化和构造方法。

加载顺序

当加载类对象时,首先初始化静态属性,然后执行静态块;当实例化对象时,首先执行构造块(直接写在类中的代码块),然后执行构造方法。至于各静态块和静态属性初始化哪个些执行,是按代码的先后顺序。属性、构造块(也就是上面的实例块)、构造方法之间的执行顺序(但构造块一定会在构造方法前执行),也是按代码的先后顺序。

  • 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、付费专栏及课程。

余额充值