构造器中绝对绝对不能调用可被覆盖的方法

代码

看下面代码示例。思考程序的输出内容

import java.time.Instant;

class Super{
    public Super(){
        overrideMe();
    }
    public void overrideMe(){

    }
}
public class Student extends Super{

    private final Instant instant;

    Student(){
        instant = Instant.now();
    }

    @Override
    public void overrideMe(){
        System.out.println(instant);
    }

    public static void main(String[] args){
        Student stu = new Student();
        stu.overrideMe();
    }
}

程序的运行结果是:

null
2020-05-13T02:51:01.398Z

思考:为什么两次调用 overrideMe() 方法只有第二次成功输出了instant对象

分析

首先我们分析一下代码,代码中有两个类,Super 类和 Student 类,Super 类中有一个 overrideMe()方法,方法体为空,Super 类的构造器中调用了 overrideMe()方法。Student 继承了 Super 类,Student 类中有一个 Instant 类型的对象,在 Student 的构造器中被实例化,同时 Student 类覆盖了 Super 类的 overrideMe()方法,在方法体中打印 instant 对象。我们在 main 函数中通过调用 Student 的构造器创建了 stu 对象,并且调用其overrideMe()方法。

继续详细的分析代码↓↓

先从main函数下手

public static void main(String[] args){
    Student stu = new Student();
    stu.overrideMe();
}

在main方法中我们通过 Student stu = new Student() 调用 Student 的构造器,由于 Student 类继承了 Super 类而 调用子类构造器之前会先调用超类的构造器,也就是在调用 Student 类的构造器前会先调用 Super 类的构造器↓↓

public Super(){
	overrideMe();
}

在超类构造器中,又调用了 overrideMe()方法,此方法在 Super 中的方法体内容为空,但是由于 overrideMe()方法在 Student 中被覆盖了,实际上 Super 类构造器调用的是 Student 类中 overrideMe()方法↓↓

@Override
public void overrideMe(){
    System.out.println(instant);
}

此方法打印 Student 中 instant 对象,但是此时 Student 的构造函数并没有被调用过,所以此此时 instant 对象实际为 null,这也就是程序运行结果中第一行打印 null 的原因。调用完此方法后,程序回到 Super 类构造器中,超类构造器执行完成,此时才会调用子类(Student)的构造器↓↓

Student(){
    instant = Instant.now();
}

在 Student 的构造方法中通过调用 Instant.now() 方法(获取当前时间戳),实例化了 Student 中的 instant 对象,至此 Student 实例化完成,回到了main 方法↓↓

public static void main(String[] args){
    Student stu = new Student();
    stu.overrideMe();
}

实例化 Student 完成之后我们又调用了其 overrideMe() 方法↓↓

@Override
public void overrideMe(){
    System.out.println(instant);
}

此时,instant 对象在刚才的 Student 实例化时也被实例化了,此时打印的结果就是我们在实例化时获取到的时间戳了,这就是第二行输出结果

回到主题,构造器中不能调用可被覆盖的方法,我们稍微改造一下 Student 中的 overrideMe()方法

@Override
public void overrideMe(){
	System.out.println(instant.getNano());
}

通过调用 instant 对象的 getNano()方法,我们意图获取时间戳的纳秒表示。通过上面的分析,在第一次调用overrideMe()方法时,instant对象为 null,所以我们可以获得下面的输出结果:

Exception in thread "main" java.lang.NullPointerException
        at Student.overrideMe(Student.java:21)
        at Super.<init>(Student.java:5)
        at Student.<init>(Student.java:15)
        at Student.main(Student.java:25)

没错肯定会出现空指针异常!

这也就是为什构造器中不能调用可被覆盖的方法,因为对于可覆盖方法,在子类覆盖过程中,很有可能会使用到子类中特有的属性,而由于实例化顺序的原因,在父类中调用被覆盖了的方法,就很有可能会造成程序运行失败。

注:通过构造器调用私有方法final 方法静态方法是安全的,因为这些方法都不能被覆盖


个人公众号

我的个人公众号已经开通了,正在按计划建设中,以后的文章会第一时间发布在公众号中,同时也给大家准备了亿点学习资料,关注公众号后台发送:学习资料

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Java ,构造方法可以相互调用。这种调用关系被称为构造方法的重载(overloading)。在一个构造方法,可以通过使用 `this()` 关键字来调用另一个构造方法。需要注意的是,`this()` 关键字必须放在构造方法的第一行,而且一个构造方法只能调用一个构造方法。 以下是一个示例代码,展示了构造方法之间相互调用的用法: ```java public class Person { private String name; private int age; public Person() { this("Unknown", 0); } public Person(String name) { this(name, 0); } public Person(int age) { this("Unknown", age); } public Person(String name, int age) { this.name = name; this.age = age; } public void introduce() { System.out.println("My name is " + name + ", and I am " + age + " years old."); } public static void main(String[] args) { Person p1 = new Person(); p1.introduce(); // 输出:My name is Unknown, and I am 0 years old. Person p2 = new Person("Tom"); p2.introduce(); // 输出:My name is Tom, and I am 0 years old. Person p3 = new Person(20); p3.introduce(); // 输出:My name is Unknown, and I am 20 years old. Person p4 = new Person("Tony", 30); p4.introduce(); // 输出:My name is Tony, and I am 30 years old. } } ``` 在上面的代码,我们定义了一个名为 Person 的类,其包含了四个构造方法。其,无参构造方法调用了一个参数为"Unknown"和 0 的构造方法;一个参数为 name 的构造方法调用了一个参数为 name 和 0 的构造方法;一个参数为 age 的构造方法调用了一个参数为"Unknown"和 age 的构造方法;最后一个参数为 name 和 age 的构造方法直接初始化了类的成员变量。 在 main() 方法,我们分别使用不同的构造方法创建了四个 Person 对象,并使用 `introduce()` 方法输出了它们的信息。可以看到,通过构造方法之间的相互调用,我们可以避免代码冗余,提高了代码的复用性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

java.util.Man

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值