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本质是不可变数据载体,编译器自动生成constructor、getter、equals、hashCode、toString。它不是“简化版class”,而是语义明确的“值对象”。
ECTouch是上海商创网络科技有限公司推出的一套基于 PHP 和 MySQL 数据库构建的开源且易于使用的移动商城网店系统!应用于各种服务器平台的高效、快速和易于管理的网店解决方案,采用稳定的MVC框架开发,完美对接ecshop系统与模板堂众多模板,为中小企业提供最佳的移动电商解决方案。ECTouch程序源代码完全无加密。安装时只需将已集成的文件夹放进指定位置,通过浏览器访问一键安装,无需对已有
- 适合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初始化顺序固定:父类静态块 → 子类静态块 → 父类实例块 → 父类构造器 → 子类实例块 → 子类构造器。若在实例块或构造器中提前调用子类重写的方法,可能访问到未初始化的字段。
立即学习“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+显式生命周期管理。









