父类构造器的隐式调用和显式调用
2016-12-14 11:59
169 查看
当调用某个类的构造器来创建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调用最多只能使用其中之一,而且最多只能调用一次.
当所有父类的非静态初始化块,构造器一次调用完成后,系统调用本类的非静态初始化块,构造器执行初始化,最后返回本类的实例.
假设有如此的类继承结构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调用最多只能使用其中之一,而且最多只能调用一次.
相关文章推荐
- 父类构造器的隐式调用和显式调用
- linux下动态链接库(.so)的显式调用和隐式调用
- java 构造方法 super 隐式调用 显式调用
- 基类和派生类的构造函数,隐式调用与显式调用
- Dll的显式和隐式调用
- Qt调用qt生成的dll 显式和隐式(静态动态)
- C++中显式和隐式调用(直接返回结构体如何生成对象)
- linux下动态链接库(.so)的显式调用和隐式调用
- 显式调用与隐式调用
- java继承之(隐式或显式)调用父类的构造函数
- Dll的显式和隐式调用
- dll文件的隐式调用和显式调用(方法示例)
- linux下动态链接库(.so)的显式调用和隐式调用
- 【牛客 题库】 求字符串长度和大小 || 基类和派生类的构造函数,隐式调用与显式调用 || 数组下标的数据类型
- linux下动态链接库(.so)的显式调用和隐式调用
- Java 构造器之构造器的隐式调用和显式调用
- 程序显式调用和隐式调用linux动态链接库(*.so)的区别。
- 动态库的显式调用和隐式调用的默认参数问题
- 关于显式调用DLL、隐式调用DLL和查看DLL输出函数
- Dll的显式和隐式调用