Java类与对象的创建与使用技巧

Java中new对象本身不抛NullPointerException,而是后续访问未初始化字段或方法所致;常见于构造函数未赋值final字段、依赖注入失败、静态工厂返回null未校验等场景。

Java中new对象时为什么报NullPointerException

不是new本身抛空指针,而是后续调用未初始化的字段或方法导致。常见于:构造函数没给final字段赋值、依赖注入失败后直接调用service.doSomething()、或静态工厂返回null却没校验。

  • 检查构造函数是否覆盖了所有必需字段,尤其带@RequiredArgsConstructor(Lombok)时,确保final字段在参数中传入
  • 避免在构造函数里调用可被子类重写的方法——子类字段此时还未初始化,容易NPE
  • Objects.requireNonNull(obj, "msg")在构造函数入口主动拦截null参数

什么时候该用static factory method代替public constructor

当需要控制实例创建逻辑、返回子类型、或避免重复构造开销时,static factory更灵活。比如Boolean.valueOf("true")复用缓存对象,而new Boolean("true")每次都新建。

  • 命名清晰:LocalDateTime.ofEpochSecond()比构造函数参数列表更易读
  • 可返回任意子类:Collections.unmodifiableList()返回包装类,非原始类型
  • 不必每次创建新对象:Optional.empty()始终返回同一个单例
  • 注意缺点:不能被继承(static方法不参与多态),且类若无public构造器,外部无法子类化

record类适合哪些场景,和普通class有什么关键区别

record本质是不可变数据载体,编译器自动生成constructorgetterequalshashCodetoString。它不是“简化版class”,而是语义明确的“值对象”。

  • 适合DTO、配置项、函数返回值等纯数据结构,例如record Point(int x, int y) {}
  • 不能有普通字段(只能有组件字段)、不能继承其他类(隐式extends java.lang.Record)、所有字段默认final
  • 若需定制toString()或添加业务方法,可以写,但别破坏不可变性;加transient字段会破坏序列化一致性,慎用
  • sealed class配合很好,用于建模有限状态,如enum Status { PENDING, DONE; } + record Success(String data) implements Result {}

对象初始化顺序错乱导致字段值不符合预期

Java初始化顺序固定:父类静态块 → 子类静态块 → 父类实例块 → 父类构造器 → 子类实例块 → 子类构造器。若在实例块或构造器中提前调用子类重写的方法,可能访问到未初始化的字段。

class Parent {
    String s = "parent";
    { System.out.println(s); } // 输出 "parent"
    Parent() { init(); }       // 调用子类重写的init()
    void init() {}
}

class Child extends Parent {
    String s = "child";         // 此时s还没赋值!
    void init() { System.out.println(s); } // 输出 null
}
  • 避免在构造器中调用this.xxx(

    )
    super.xxx()以外的非final方法
  • 把依赖初始化的逻辑移到private final方法中,并在构造器末尾调用
  • 使用IDEA的“Constructor calls overridable method”检查,或启用-Xlint:serial等编译警告
复杂点在于初始化时机与继承边界交织,哪怕只改一行final修饰符,也可能让原本安全的代码在子类中崩掉。别信“我只用record就万事大吉”,遇到要延迟加载、缓存计算、或跨线程共享对象时,还是得回到普通class+显式生命周期管理。