Java构造方法是对象创建时唯一执行的特殊方法;未定义时编译器自动插入无参默认构造,但定义任一构造后默认构造立即消失;必须与类名完全一致且无返回类型,否则视为普通方法;多个构造间用this()调用须为首行且唯一;构造中调用非final实例方法可能导致子类字段未初始化异常;继承有参父类时须显式调用super()。
Java 构造方法不是“可选的初始化方式”,而是对象创建时唯一能执行的、与类同名的特殊方法;没写任何构造方法时,编译器会自动插入一个无参默认构造方法;但只要定义了任意一个构造方法(哪怕带参数),这个默认构造方法就**立即消失**。
这是最常被误写的点:加 void 或其他返回类型会让它变成普通方法,JVM 不再识别为构造方法,导致 new 实例时报 NoSuchMethodError 或编译失败。
常见错误现象:
public void Person() { ... } → 编译通过但不是构造方法,new Person() 会调用不到,报错Person,却写成 person() → 同样被当普通方法正确写法只有一种形式:
public class Person {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
}
this(...) 必须是构造方法的第一条语句,且只能出现一次;它用于复用已有构造逻辑,避免重复赋值。但不能和 super(...) 共存,也不能在普通方法里调用。
使用场景:
错误示例(编译不通过):
public Person(String name) {
System.out.println("before"); // ❌ this() 必须第一行
this(name, 0);
}
正确写法:
public Person(String name) {
this(name, 0); // ✅ 第一行,且只出现一次
}
如果在构造方法里调用了被子类重写的方法(且该方法不是 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 完成后再手动调用当类继承自有参构造的父类(如 extends Exception),子类若没显式写 super(...),编译器会尝试插 super() —— 但父类根本没有无参构造,直接编译失败。
典型错误信息:
Implicit super constructor Exception() is undefined. Must explicitly invoke another constructor
解决办法:
super(message) 或对应参数
这问题在写自定义异常或框架扩展类时高频出现,尤其容易在重构父类后突然爆发。