继承中构造方法的关系

之前一直在讲继承中我们不能继承父类的构造方法,但是我们可以调用父类的构造方法,那我们到底如何去让子类调用父类的构造方法呢?之前一直没有介绍super和this关键字的用法,在我们学习了this和super关键字的用法之后,我们就来了解一下子类如何调用父类的构造方法的。


继承中构造方法的关系

 

  1. 在初始化子类数据之前,必须先完成对父类数据的初始化(因为在初始化子类数据的时候,可能会使用到父类中的数据,所以必须先把父类数据准备好)
  2. 如果在子类的构造方法中,没有显式的调用任何其他构造方法(本类、父类),在构造方法的第一句,系统默认给你加上super(),默认要访问父类的空参构造
  3. 如果在子类的构造方法中,手动增加了访问父类的构造方法,那么系统将不给你增加任何构造方法的访问
  4. 如果在子类的构造方法中,手动增加了访问本类的其他构造方法,那么系统将不给你增加任何构造方法的访问
  5. 构造方法不能递归调用
  6. 构造方法不能自己直接访问自己,也不能自己间接的访问自己

总结
【目标】子类的构造方法,一定要先访问父类的构造方法
【特点】:

  •      【this语句】和【super语句】必须在构造方法的第一行
  •       super语句在第一句是为了保证父类的构造方法必须先执行
  •       this语句在第一句是因为间接的访问了super,super必须先执行
  •       在构造方法中,【this语句】和【super语句】不能共存
  •       this语句和super语句都要在构造方法的第一行
  •     【this语句】和【super语句】不能出现在其他非构造方法的非静态成员方法中
  •       其他方法都是在对象创建之后,才能调用的方法,如果能调用this语句,就相 当于间接的使用对象,调用构造方法


     


我们用代码来验证上述关系:

public class 继承中构造方法的关系 {
	public static void main(String[] args) {
		Zii z = new Zii();
		Zii z1 = new Zii(12);
		z1.show();
	}
}

class Fuu{
	private int age;
	
	public Fuu() {
		System.out.println("父类的无参构造被调用了");
	}
	
	public Fuu(int age) {
		this.age = age;
		System.out.println("父类的有参构造方法被调用了");
	}
	
	public void show() {
		System.out.println("父类的成员方法被调用了,age="+age);
	}
}
class Zii extends Fuu{
	public Zii() {
		System.out.println("子类的无参构造方法被调用了");
	}
	public Zii(int age) {
		System.out.println("子类的有参构造方法被调用了");
	}
}

 输出结果:

父类的无参构造被调用了
子类的无参构造方法被调用了
父类的无参构造被调用了
子类的有参构造方法被调用了
父类的成员方法被调用了,age=0

从结果我们可以看出:

①我们在子类的实例化中,子类的构造方法并没有有明确的代码写入调用父类的构造方法,但是结果显示,在每一个子类对象实例化的同时都有父类的无参构造方法被调用,这就验证了上面所说的第一条,必须先加载父类才会加载子类。

②super()会调用父类的无参构造方法,我们没有写入,但是系统给我们提供了一个默认的调用,所以我们把子类中的构造方法加上super()也会的到同样的效果(修改父类代码):

class Zii extends Fuu{
	public Zii() {
		super();
		System.out.println("子类的无参构造方法被调用了");
	}
	public Zii(int age) {
		super();
		System.out.println("子类的有参构造方法被调用了");
	}
}

输出:

父类的无参构造被调用了
子类的无参构造方法被调用了
父类的无参构造被调用了
子类的有参构造方法被调用了
父类的成员方法被调用了,age=0

③:我们再来修改子类的有参构造方法,加入this();调用子类的无参构造。(修改父类代码)

class Zii extends Fuu{
	public Zii() {
		super();
		System.out.println("子类的无参构造方法被调用了");
	}
	public Zii(int age) {
		this();
		System.out.println("子类的有参构造方法被调用了");
	}
}

输出结果:

父类的无参构造被调用了
子类的无参构造方法被调用了
父类的无参构造被调用了
子类的无参构造方法被调用了
子类的有参构造方法被调用了
父类的成员方法被调用了,age=0

我们发现:我们只写入了this(),也没有使用super(),但是还是显示调用了父类的无参构造方法。所以说,this中包含了对super的调用。

所以我们可以得出总结:   不管子类有没有写入调用父类构造方法的代码,子类对象的创建一定会调用父类的构造方法。

④我们发现,最后的输出结果age一直等于0,而我们在构造的时候一直都是调用父类的无参构造方法,我们来尝试调用父类的有参构造方法(修改父类代码):

class Zii extends Fuu{
	public Zii() {
		super();
		System.out.println("子类的无参构造方法被调用了");
	}
	public Zii(int age) {
		super(age);
		System.out.println("子类的有参构造方法被调用了");
	}
}

结果输出:

父类的无参构造被调用了
子类的无参构造方法被调用了
父类的有参构造方法被调用了
子类的有参构造方法被调用了
父类的成员方法被调用了,age=12

我们发现age有了我们赋予的值,这就是父类有参构造方法的调用。

同时,我们还发现了在调用有参构造方法的时候,没有输出关于父类无参构造方法的语句,就又印证了上面所说的第三条,当调用了一个构造方法的时候,就不会调用另外的构造方法,也就是要么调用无参构造,要么调用有参构造,不能多选,系统也不会自动重复。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值