父类构造器的隐式调用和显式调用

当调用某个类的构造器来创建Java对象时,系统总会先调用父类的非静态初始化块进行初始化.这个调用是隐式执行的,而且父类的静态初始化块总是会被执行.接着会调用父类的一个或多个构造器执行初始化,这个调用既可以是通过super进行显式调用,也可以是隐式调用.

当所有父类的非静态初始化块,构造器一次调用完成后,系统调用本类的非静态初始化块,构造器执行初始化,最后返回本类的实例.

假设有如此的类继承结构Object<--Parent<--Mid<--Sub.那么,对于这样的类继承结构(Java程序里所有类的最终父类都是java.lang.Object类),如果在程序中创建Sub对象,程序会按如下步骤进行初始化.

1)执行Object类非静态初始化块(如果有的话).

2)隐式或显式调用Object类的一个或多个构造器执行初始化.

3)执行Parent类非静态初始化块(如果有的话).

4)隐式或显式调用Parent类的一个或多个构造器执行初始化.

5)执行Mid类非静态初始化块(如果有的话).

6)隐式或显式调用Mid类的一个或多个构造器执行初始化.

7)执行Sub类非静态初始化块(如果有的话).

8)隐式或显式调用Sub类的一个或多个构造器执行初始化.

下面程序演示了创建Java对象时的初始化过程

package com.lic.array;

public class Demo13 {

	public static void main(String[] args) {
		new Wolf(2.3);
	}
}
class Creature{
	{
		System.out.println("Creature的非静态初始化块");
	}
	// 下面定义两个构造器
	public Creature(){
		System.out.println("Creature无参数构造器");
	}
	public Creature(String name){
		// 使用this调用另一个重载,无参数的构造器
		this();
		System.out.println("Creature带有一个参数的构造器,name参数:"+name);
	}
}
class Animal extends Creature{
	{
		System.out.println("Animal的非静态初始化块");
	}
	public Animal(String name){
		super(name);
		System.out.println("Animal带一个参数的构造器,name参数:"+name);
	}
	public Animal(String name, int age){
		// 使用this调用另一个重载的构造器
		this(name);
		System.out.println("Animal带2个参数的构造器,其中age:"+age);
	}
}
class Wolf extends Animal{
	{
		System.out.println("Wolf的非静态初始化块");
	}
	public Wolf(){
		// 显式调用父类有2个参数的构造器
		super("灰太狼",3);
		System.out.println("Wolf无参数的构造器");
	}
	public Wolf(double weight){
		// 使用this调用另一个重载的构造器
		this();
		System.out.println("Wolf的带weight参数的构造器,weight参数:"+weight);
	}
}
/**
 * 子类构造器会显式或隐式的调用父类构造器,默认调用无参构造器.如果父类定义了有参构造器且未定义无参构造器,则必须显式调用父类已经声明的构造器.
*/
控制台输出如下:

Creature的非静态初始化块
Creature无参数构造器
Creature带有一个参数的构造器,name参数:灰太狼
Animal的非静态初始化块
Animal带一个参数的构造器,name参数:灰太狼
Animal带2个参数的构造器,其中age:3
Wolf的非静态初始化块
Wolf无参数的构造器
Wolf的带weight参数的构造器,weight参数:2.3
上面程序定义了Creature,Animal,Wolf这3个类,其中Animal是Creature的子类,Wolf是Animal的子类.3个类都包含了非静态初始化块,构造器成分.当程序调用Wolf的指定构造器创建Wolf实例时,程序会按上面介绍的初始化步骤执行初始化.执行上面程序,结果如上图所示.

只要在程序创建Java对象,系统总是先调用最顶层父类的初始化操作,包括吃实话块和构造器,然后依次向下调用所有父类的初始化操作,最终执行本类的初始化操作返回本类的实例.至于调用父类的哪个构造器执行初始化,则分为如下几种情况:

1)子类构造器执行体的第一行代码使用super显式调用父类构造器,系统根据super调用里传入的实参列表来确定调用父类的哪个构造器;

2)子类构造器执行体的第一行代码使用this显式调用本类中重载的构造器,系统将根据this调用里传入的实参列表来确定本类的另一个构造器(执行本类中另一个构造器时即进入第一种情况);

3)子类构造器执行体中既没有super调用,也没有this调用,系统将会在执行子类构造器之前,隐式调用父类无参数的构造器.

super调用用于显式调用父类的构造器,this调用用于显式调用本类中的另一个重载的构造器.super调用和this调用都只能在构造器中使用,而且super调用和this调用都必须作为构造器的第一行代码,因此构造器中的super调用和this调用最多只能使用其中之一,而且最多只能调用一次.



  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Zerlinda_Li

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

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

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

打赏作者

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

抵扣说明:

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

余额充值