关于java中代码块和构造函数的加载顺序以及父类和子类中的加载顺序

本文通过实例演示了Java中构造方法、静态代码块、构造代码块及普通代码块的加载顺序,并展示了父类与子类构造过程中的执行流程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

首先,我们简单了解一下代码块和构造方法

静态代码块:

存在类内方法外,使用static修饰,作用是类的预加载,当类第一次被加载的时候执行一次,优先于构造代码块 优先于构造方法
构造代码块:

存在类内方法外,作用是对象的预加载,每当new调用构造方法的时候 会优先执行构造代码块,每new一次就优先执行一次构造代码块

普通代码块:

存在成员方法中,作用是控制变量的生命周期,执行完后立即消失.
构造方法:

创建对象的方式,每当调用构造方法时,构造方法执行一次 

关于加载顺序,我们将通过实现代码来查看

一、定义一个父类

public class Die {
	int age;

	static {
		System.out.println("父类的静态代码块");
	}

	{
		// 父类的构造代码块
		System.out.println("父类的构造代码块");
	}

	// 父类无参
	public Die() {
		this(20);
		System.out.println("父类的无参构造方法");
		// 普通代码块
		{
			System.out.println("父类的代码块");
		}
	}

	// 父类含参
	public Die(int age) {
		super();
		this.age = age;
		System.out.println("父类的含参构造方法");

	}

}

二、定义一个子类

public class Zi extends Die{
	String name;
	static{
		System.out.println("子类的静态代码块");
	}
	
	{
		//构造代码块
		System.out.println("子类的构造代码块");
	}
	
	//子类无参
	public Zi() {
		this("");//在执行子类的含参构造方法 但是没有对象产生
		System.out.println("子类无参构造方法");
		//普通代码块
		{
			System.out.println("子类的代码块");
		}
	}

	//子类含参
	public Zi(String name) {
		super();
		this.name = name;
		System.out.println("子类的含参构造");
	}
	
}

三、测试类

public class ExtendsDemo {
	public static void main(String[] args){
		Zi z = new Zi();
		System.out.println("==========");
		Die d = new Zi();
	}
}

结果:

Zi z = new Zi()    的运行结果:
父类的静态代码块
子类的静态代码块
父类的构造代码块
父类的含参构造方法
父类的无参构造方法
父类的代码块
子类的构造代码块
子类的含参构造
子类无参构造方法
子类的代码块
Die d = new Zi()    的运行结果:
父类的构造代码块
父类的含参构造方法
父类的无参构造方法
父类的代码块
子类的构造代码块
子类的含参构造
子类无参构造方法
子类的代码块

静态代码块只加载一次,所以第二次没有加载静态代码块

<think>首先,用户的问题是关于Java子类实例化时父类构造器方法的执行顺序。具体来说,用户问:"Java 子类实例化时 父类构造方法执行顺序 方法调用顺序"。我需要基于系统指令提供的引用来回答。 系统指令要求: - 所有行内数学表达式用$...$格式(但这个问题可能不涉及数学表达式)。 - 独立公式用$$...$$格式(同样,可能不适用)。 - LaTeX语法正确。 - 使用中文回答。 - 生成相关问题。 - 回答中引用的段落末尾自然地添加引用标识,例如[^1]。 - 回答结构清晰,帮助用户逐步解决问题。 - 保证回答真实可靠。 用户提供的引用包括: - [^1]: 子类调用父类构造方法,Java虚拟机先执行父类构造方法,再执行子类构造方法。 - [^2]: 执行顺序:父类静态代码块 -> 子类静态代码块 -> 父类构造代码块 -> 父类构造方法 -> 子类构造代码块 -> 子类构造方法。 - [^3]: 子类初始化时调用父类构造方法,通常有隐式的super()调用父类无参数构造方法。 - [^4]: 详细示例显示执行顺序,包括构造函数、属性、方法的初始化。 用户的上一个问题似乎是关于子类实例化时的顺序,所以我需要综合这些引用来构建回答。 回答结构: 1. **逐步解释执行顺序**:基于引用[^2][^4],列出完整的执行顺序。 2. **关键点**: - 父类构造方法先执行。 - 静态代码块优先于实例初始化。 - 构造代码块在构造方法之前。 - 方法调用顺序取决于实例类型。 3. **使用示例**:提供一个简单的代码示例来演示顺序。 4. **生成相关问题**:在结尾添加相关问题,如指令要求。 完整执行顺序(基于引用[^2]): - 父类静态代码块 - 子类静态代码块 - 父类构造代码块 - 父类构造方法 - 子类构造代码块 - 子类构造方法 在方法调用中,如果子类覆盖了方法,调用时会使用子类的方法版本(多态)。 回答应清晰、结构化: - 引言:简述顺序。 - 详细步骤:分步解释。 - 示例:提供代码输出。 - 总结:强调关键点。 - 相关问题:添加相关问题。 相关问题生成(基于指令示例): 1. Java中静态代码块实例初始化块的区别? 2. 如何显式调用父类构造方法? 3. 多态在方法调用中如何工作? 4. 如果父类没有无参数构造方法,子类如何处理? 确保引用自然添加:在解释中,当提到特定点时,添加引用标识,如[^2]。 最终回答用中文。</think>### Java子类实例化时父类构造方法方法的执行顺序 在Java中,当创建子类实例时,父类子类的构造方法、初始化块(包括静态实例块)以及方法调用遵循特定的顺序。这个顺序由Java的类加载初始化机制决定,确保对象从父类继承的成员被正确初始化。下面我将逐步解释这个顺序,并提供一个代码示例来演示。整个过程基于Java的类加载初始化规则,确保可靠性准确性。 #### 关键执行顺序总结 当实例化子类时,完整的执行顺序如下(从类加载到对象初始化): 1. **静态初始化**:优先执行父类的静态代码块,然后执行子类的静态代码块(只在类首次加载时执行一次)。 2. **父类实例初始化**:执行父类的实例初始化块(构造代码块),然后执行父类的构造方法。 3. **子类实例初始化**:执行子类的实例初始化块(构造代码块),然后执行子类的构造方法。 4. **方法调用**:如果子类覆盖了父类的方法,在调用时会基于对象的实际类型(子类)执行多态行为。 完整的顺序链为: 父类静态代码块子类静态代码块父类构造代码块父类构造方法 → 子类构造代码块子类构造方法[^2]。 #### 详细步骤解释 1. **静态初始化阶段(类加载时)**: - Java虚拟机(JVM)加载类时,先加载父类,执行其静态代码块(`static { ... }`)。 - 然后加载子类,执行其静态代码块。静态代码块只执行一次,无论创建多少实例。 2. **父类实例初始化阶段**: - 当通过`new SubClass()`创建子类对象时,JVM首先隐式调用父类的构造方法(通常通过`super()`,默认调用父类无参构造方法)。 - 在父类构造方法执行前,先执行父类的实例初始化块(即构造代码块,`{ ... }`)。 - 接着执行父类的构造方法(例如`Parent() { ... }`),初始化父类成员变量。 3. **子类实例初始化阶段**: - 父类初始化完成后,JVM执行子类的实例初始化块(构造代码块)。 - 然后执行子类的构造方法(例如`Child() { ... }`),初始化子类成员变量。 4. **方法调用顺序**: - 在构造方法或初始化块中调用方法时,如果子类覆盖了父类的方法,则调用子类版本(多态)。 - 例如,在父类构造方法中调用一个被覆盖的方法,会执行子类的实现,但此时子类成员可能尚未初始化(导致潜在问题,如`null`值)[^4]。 #### 代码示例演示 以下是一个简单的Java程序,展示子类实例化时的执行顺序。父类`Parent`子类`Child`分别包含静态代码块、实例初始化块、构造方法覆盖的方法。 ```java class Parent { static { System.out.println("父类静态代码块"); // 步骤1 } { System.out.println("父类实例初始化块"); // 步骤3 } public Parent() { System.out.println("父类构造方法"); // 步骤4 print(); // 调用方法,多态行为 } public void print() { System.out.println("父类print方法"); } } class Child extends Parent { static { System.out.println("子类静态代码块"); // 步骤2 } { System.out.println("子类实例初始化块"); // 步骤5 } private String name = "Child"; // 子类成员变量 public Child() { System.out.println("子类构造方法"); // 步骤6 print(); // 调用方法 } @Override public void print() { System.out.println("子类print方法: name = " + name); // 多态调用 } } public class Main { public static void main(String[] args) { new Child(); // 创建子类实例 } } ``` **输出结果**: ``` 父类静态代码块 子类静态代码块 父类实例初始化块 父类构造方法 子类print方法: name = null // 注意:此时子类name未初始化 子类实例初始化块 子类构造方法 子类print方法: name = Child // 子类成员已初始化 ``` **输出解释**: - 静态代码块先执行(父类子类),只执行一次。 - 父类实例初始化块构造方法先于子类执行。 - 在父类构造方法中调用`print()`时,由于多态,执行子类的`print()`方法,但此时子类的`name`尚未初始化(为`null`),显示"name = null"。 - 子类构造方法执行后,`name`被初始化,调用`print()`时显示正确值。 #### 关键注意事项 - **隐式`super()`调用**:如果子类构造方法没有显式调用`super(...)`或`this(...)`,编译器会自动添加`super()`调用父类无参构造方法[^3]。 - **多态行为**:方法调用基于对象的运行时类型(子类),但构造方法中调用方法时,子类成员可能未初始化,需避免。 - **继承链**:在多级继承中,顺序从最顶层父类开始,逐级向下执行[^1]。 - **静态成员**:静态代码块变量在类加载时初始化,优先于实例创建。 通过这个顺序,Java确保父类成员被正确初始化后,再处理子类逻辑,避免数据不一致[^4]。如果您有具体代码场景,我可以进一步分析。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值