今天做了一道Java题,人都傻了,好好的super感觉都快被玩坏了。虽然博主并不清楚究竟是什么原因导致的,但是可以肯定的是,我需要把它写出来。
这里就不详细描述问题,直接上代码
首先是一个Person类
class Person {
public Person() {
System.out.println("调用父类构造方法");
System.out.println("在父类构造方法中调用func()方法");
func();
}
public void func() {
System.out.println("调用父类func()");
}
}
这个是一个稀松平常的Person类,在构造函数里输出两句话,调用一个func()方法 。
然后是重点,一个Person的子类
public class Teacher extends Person {
public Teacher() {
super();
System.out.println("调用子类构造方法");
}
public void func() {
//super.func();
System.out.println("调用子类func()");
}
public static void main(String[] args) {
Teacher t1 = new Teacher();
}
}
这个子类请先无视注释的部分, 这个Teacher类由三部分组成,一个构造方法,调用了super();一个func()方法,用以重写父类的func()方法;一个main方法,用以执行程序。
接下来,让我们猜一下,执行结果是什么?正常思路应该是:先调用子类构造函数,然后执行super()调用父类构造函数,父类构造函数里面执行func()方法,然后返回子类构造方法将构造方法执行完毕。
所以说,问题的关键在哪呢,问题的关键在于这个func()方法是子类的,super()去调用父类构造函数,结果这个父类构造函数执行的确是子类的func(),这是什么鬼?恕博主才疏学浅,确是不是很理解super这玩意,所以不明所以。
这是完整的代码
class Person {
public Person() {
System.out.println("调用父类构造方法");
System.out.println("在父类构造方法中调用func()方法");
func();
}
public void func() {
System.out.println("调用父类func()");
}
}
public class Teacher extends Person {
public Teacher() {
super();
System.out.println("调用子类构造方法");
}
public void func() {
//super.func();
System.out.println("调用子类func()");
}
public static void main(String[] args) {
Teacher t1 = new Teacher();
}
}
结果在这
调用父类构造方法
在父类构造方法中调用func()方法
调用子类func()
调用子类构造方法
于是,到此处我可以得出一个结论,被重写的方法不会被super调用。可是,如果仅仅如此,我就不必困惑了。
我前面提了一嘴,上面代码注释的地方。它既然是调用子类的func(),那么我在子类func()中在调用父类的func()呗。
将上面注释的话恢复后,结果如下:
调用父类构造方法
在父类构造方法中调用func()方法
调用父类func()
调用子类func()
调用子类构造方法
百度到super的作用主要在下面三种情况下:
1、调用父类被子类重写的方法;
2、调用父类被子类重定义的字段(被隐藏的成员变量);
3、调用父类的构造方法;
故在此做一个猜想,这三种方法应该是不可以嵌套使用的(如果可以的话,上面的例子就不应该调用子类的func()),如果super调用的父类被子类重写的方法里面使用了被子类重新定义的字段,那么被子类重写的方法可以成功调用,但是里面所使用的被重新定义的字段,依旧是子类的。