Java构造器重载与默认构造函数的使用

是的,但仅限于类中未定义任何构造器时;一旦定义了任意构造器,编译器便不再生成默认无参构造函数,且其访问修饰符与类一致,等价于public A(){super();}。

Java中没有显式定义构造器时,编译器会自动生成默认构造函数吗?

是的,但仅限于「类中未定义任何构造器」的情况。一旦你写了哪怕一个带参数的构造器,编译器就不再生成无参的默认构造函数。

这意味着:如果其他代码(比如框架反射、序列化工具、或子类 super() 调用)依赖无参构造器,而你只写了 public Person(String name),就会直接报错 java.lang.NoSuchMethodException: Person.()

  • 默认构造函数一定是无参、访问修饰符与类一致(如 public class A → 默认构造器为 public A()
  • 它不执行任何逻辑,等价于 public A() { super(); }
  • 继承链中,若父类删掉了无参构造器,子类必须显式调用 super(...),否则编译失败

构造器重载时,如何避免 this() 和 super() 调用冲突?

Java 规定:每个构造器的第一条语句,必须是 this(...)super(...)(隐式 super() 也算)。二者不能共存,也不能出现在第二行。

常见错误是想“先初始化字段再委托”,比如:

public Person(String name) {
    this.name = name; // ❌ 编译错误:this() 必须是第一行
    this();
}

正确做法是把通用初始化逻辑提取到私有方法,或让所有构造器都通过一个“主构造器”统一入口:

public class Person {
    private String name;
    private int age;

    public Person() {
        this("unknown", 0);
    }

    public Person(String name) {
        this(name, 0);
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
}

为什么 Lombok 的 @AllArgsConstructor 不会覆盖默认构造器?

因为 Lombok 的注解是「按需生成」:@AllArgsConstructor 只生*参构造器;@NoArgsConstructor 才生成无参构造器;@RequiredArgsConstructor 生成含 @NonNull 字段的构造器。它们互不干扰。

但要注意:如果你同时加了 @AllArgsConstructor 和手动写了无参构造器,不会冲突;可一旦用了 @RequiredArgsConstructor 且字段有 @NonNull,又没配 @NoArgsConstructor,那依然没有无参构造器——这点在 Spring Bean 初始化或 JSON 反序列化(如 Jackson)时容易爆 InstantiationException

  • Spring Boot 默认要求组件类有无参构造器(除非用构造器注入 + @Autowired 显式标注)
  • Jackson 反序列化 POJO 时,默认走无参构造器 + setter;若关闭 MapperFeature.REQUIRE_SETTERS_FOR_GETTERS,则必须有无参构造器
  • Lombok 生成的构造器访问修饰符默认是 public,可通过 @AllArgsConstructor(access = AccessLevel.PROTECTED) 调整

子类构造器中,super() 调用不写会怎样?

如果父类只有带参构造器,而子类构造器里没写 super(...),也没写 this(...),编译器会尝试插入隐式 super() —— 但此时父类根本没有无参构造器,直接编译失败。

例如:

class Animal {
    public Animal(String type) { ... }
}

class Dog extends Animal {
    public Dog(String name) {
        // ❌ 编译错误:找不到 Animal()
    }
}

修复方式只有两种:

  • Dog 构造器第一行显式写 super("dog")
  • 或在 Animal 中补一个无参构造器(public Animal() { this("unknown"); }

这个限制不是运行时问题,而是编译期强制检查,容易被忽略,尤其在重构父类时删掉无参构造器却忘了同步改子类。