Java中new触发类加载、堆分配、默认初始化、构造器执行;对象销毁需满足无GC Roots可达、被判定不可达且经GC标记,非立即回收。
Java 中 new 不只是语法糖,它会依次触发类加载(若未加载)、堆内存分配、默认字段初始化、构造器执行。关键点在于:对象实际在堆上分配,但 JIT 可能通过逃逸分析做栈上分配优化(对开发者透明,但影响 GC 压力)。
常见误区是认为 new 后立即“可用”——其实若构造器抛异常(如 NullPointerException 或自定义异常),对象创建就失败,JVM 会清理已分配内存,不会留下半初始化对象。
LocalDateTime.now())比直接 new 更灵活,也便于返回缓存实例或子类new String())会快速填充年轻代,触发 Minor GCfinalize() 自 Java 9 起标记为 @Deprecated,Java 18 彻底移除。它不可靠(不保证何时执行、甚至不保证执行)、性能差、易导致对象复活(resurrection),且与现代 GC 算法(如 ZGC、Shenandoah)不兼容。
真正需要资源清理(如关闭文件句柄、释放 JNI 内存)时,应优先使用 try-with-resources;若必须异步清理,用 Cleaner:
private static final Cleaner cleaner = Cleaner.create();
private static class State implements Runnable {
private final FileDescriptor fd;
State(FileDescriptor fd) { this.fd = fd; }
public void run() { close(fd); }
}
private final Cleaner.Cleanable cleanable;
public Resource(FileDescriptor fd) {
this.cleanable = cleaner.register(this, new State(fd));
}
Cleaner 基于 PhantomReference,不阻止 GC,无复活风险Cleaner 的 run() 中执行耗时操作(如网络调用),它运行在专用线程池中cleanable.clean() 可提前触发清理,适用于确定性释放场景对象被回收不是因为“不再使用”,而是满足三个条件:没有 GC Roots 可达路径、被判定为不可达、且经过至少一次 GC 标记阶段(取决于 GC 算法)。即使满足,也不代表立即回收——ZGC 的回收是并发的,G1 的 Mixed GC 是分批次的。
典型误判场景:

public static List cache = new ArrayList(); )长期持有对象引用,造成内存泄漏remove(),导致线程结束时对象仍被持有(尤其在线程池中)调用 System.gc() 仅向 JVM 发出“建议”执行 Full GC,但 HotSpot 默认忽略该请求(除非启动参数加 -XX:+ExplicitGCInvokesConcurrent)。生产环境主动调用它,往往掩盖了真正的内存问题,还可能引发 STW 暂停。
真正需要干预 GC 的情况极少,更合理的做法是:
jstat -gc 观察 GC 频率与停顿时间-Xlog:gc*:file=gc.log 开启 GC 日志,定位晋升失败(Promotion Failure)或元空间溢出(Metaspace OOM)对象生命周期管理的核心不在“怎么杀”,而在“谁在持有着它”。排查时优先检查引用链,而不是盯着构造和 finalize。