Java构造方法是对象创建时唯一执行的特殊方法;未定义时编译器自动插入无参默认构造,但定义任一构造后默认构造立即消失;必须与类名完全一致且无返回类型,否则视为普通方法;多个构造间用this()调用须为首行且唯一;构造中调用非final实例方法可能导致子类字段未初始化异常;继承有参父类时须显式调用super()。

Java 构造方法不是“可选的初始化方式”,而是对象创建时唯一能执行的、与类同名的特殊方法;没写任何构造方法时,编译器会自动插入一个无参默认构造方法;但只要定义了任意一个构造方法(哪怕带参数),这个默认构造方法就**立即消失**。
构造方法必须与类名完全一致且无返回类型
这是最常被误写的点:加 void 或其他返回类型会让它变成普通方法,JVM 不再识别为构造方法,导致 new 实例时报 NoSuchMethodError 或编译失败。
常见错误现象:
- 写了
public void Person() { ... }→ 编译通过但不是构造方法,new Person()会调用不到,报错 - 大小写不一致,如类叫
Person,却写成person()→ 同样被当普通方法
正确写法只有一种形式:
立即学习“Java免费学习笔记(深入)”;
public class Person {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
}
多个构造方法之间用 this() 调用需满足严格条件
this(...) 必须是构造方法的第一条语句,且只能出现一次;它用于复用已有构造逻辑,避免重复赋值。但不能和 super(...) 共存,也不能在普通方法里调用。
使用场景:
- 提供多种初始化入口(如支持姓名+年龄、仅姓名、无参)
- 统一字段校验或默认值设置逻辑,集中到一个“主构造”中
错误示例(编译不通过):
public Person(String name) {
System.out.println("before"); // ❌ this() 必须第一行
this(name, 0);
}
正确写法:
public Person(String name) {
this(name, 0); // ✅ 第一行,且只出现一次
}
构造方法中调用非 final 实例方法可能导致子类状态未初始化
如果在构造方法里调用了被子类重写的方法(且该方法不是 final 或 private),JVM 会按运行时类型去调用子类版本——但此时子类字段尚未初始化,可能返回 null 或默认值(如 0、false)。
例如:
class Parent {
public Parent() {
init(); // 实际调用 Child.init(),但 Child.name 还没赋值
}
void init() { }
}
class Child extends Parent {
String name = "Alice";
void init() {
System.out.println(name.length()); // NullPointerException
}
}
规避方式:
- 构造方法内只调用
private或final方法 - 把初始化逻辑拆到独立的
init()方法,在new完成后再手动调用
IDE 自动生成构造方法时容易忽略父类构造签名
当类继承自有参构造的父类(如 extends Exception),子类若没显式写 super(...),编译器会尝试插 super() —— 但父类根本没有无参构造,直接编译失败。
典型错误信息:
Implicit super constructor Exception() is undefined. Must explicitly invoke another constructor
解决办法:
- 手动补上
super(message)或对应参数 - 在 IDE 生成构造方法时,勾选“Delegate to constructor of superclass”选项(IntelliJ/ Eclipse 均支持)
这问题在写自定义异常或框架扩展类时高频出现,尤其容易在重构父类后突然爆发。










