ChatGPT解决这个技术问题 Extra ChatGPT

Java 什么时候调用 finalize() 方法?

我需要知道何时在 JVM 中调用 finalize() 方法。我创建了一个测试类,当通过覆盖它调用 finalize() 方法时它会写入文件。它不被执行。谁能告诉我它没有执行的原因?

顺便说一句:finalize 在 Java 9 中被标记为已弃用
如果有任何东西引用了您的对象甚至类。 finalize() 并且垃圾收集没有任何影响。
从 Java9 开始不推荐使用它。 docs.oracle.com/javase/9/docs/api/java/lang/…
openjdk.java.net/jeps/421 (Java 18) 开始,终结器尚未删除。

J
Joachim Sauer

finalize 方法在对象即将被垃圾回收时调用。这可以在它有资格进行垃圾收集之后的任何时间。

请注意,一个对象完全有可能永远不会被垃圾收集(因此永远不会调用 finalize)。这可能发生在对象永远不符合 gc 条件(因为它在 JVM 的整个生命周期内都可以访问)或者在对象变得符合条件和 JVM 停止运行之间实际上没有运行垃圾收集时(这通常发生在简单的测试程序)。

有一些方法可以告诉 JVM 在尚未调用它的对象上运行 finalize,但使用它们也不是一个好主意(该方法的保证也不是很强)。

如果您依赖 finalize 来正确运行您的应用程序,那么您做错了什么。 finalize用于清理(通常是非 Java)资源。 完全正确,因为 JVM 不保证在任何对象上都会调用 finalize


@拉杰什。不,这不是“寿命”问题。您可以将程序置于无限循环中(多年),如果不需要垃圾收集器,它将永远不会运行。
@VikasVerma:完美的替代品什么都不是:你不应该需要它们。唯一有意义的情况是,如果您的类管理一些外部资源(如 TCP/IP 连接、文件……Java GC 无法处理的任何东西)。在这些情况下,Closable 接口(及其背后的想法)可能就是您想要的:使 .close() 关闭/丢弃资源并要求您的类的用户在正确的时间调用它。您可能想要添加一个“只是为了保存”的 finalize 方法,但这更像是一个调试工具而不是实际修复(因为它不够可靠)。
@Dragonborn:这实际上是一个完全不同的问题,应该单独提出。有关闭钩子,但如果 JVM 意外关闭(又名崩溃),则无法保证它们。但是它们的保证比终结器的保证要强得多(而且它们也更安全)。
您的最后一段说仅将其用于清理资源,即使不能保证它会被调用。这种乐观是在说话吗?我会假设一些不可靠的东西也不适合清理资源。
终结器有时可以挽救这一天......我有一个案例,第 3 方库正在使用 FileInputStream,从不关闭它。我的代码正在调用库的代码,然后尝试移动文件,但由于它仍处于打开状态而失败。我不得不强制调用 System.gc() 来调用 FileInputStream::finalize(),然后我才能移动文件。
P
Paul Bellora

一般来说,最好不要依赖 finalize() 进行任何清理等。

根据 Javadoc(值得一读),它是:

当垃圾收集确定不再有对该对象的引用时,由对象上的垃圾收集器调用。

正如 Joachim 所指出的,如果对象总是可访问的,那么这在程序的生命周期中可能永远不会发生。

此外,垃圾收集器不能保证在任何特定时间运行。一般来说,我想说的是 finalize() 可能不是一般使用的最佳方法,除非您需要它特定的东西。


换句话说(只是为了澄清未来的读者)它永远不会在主类上调用,因为当主类关闭时,不需要收集垃圾。操作系统会清理应用程序使用的所有内容。
“不是最好的使用方法......除非你需要它特定的东西” - 呃,这句话适用于100%的一切,所以没有帮助。 Joachim Sauer 的回答要好得多
@Zom-B您的示例有助于澄清,但只是为了迂腐,如果主类创建一个非守护线程然后返回,大概可以在主类上调用它?
那么在哪些情况下 finalize 有用呢?
@MarkJeronimus - 实际上,这无关紧要。当 >>instance<< 时调用主类的 finalize() 方法。类的垃圾收集,而不是在 main 方法终止时。此外,主类可能在应用程序完成之前被垃圾收集;例如,在“主”线程创建其他线程然后返回的多线程应用程序中。 (实际上,需要一个非标准的类加载器......)
X
XpiritO

protected void finalize() throws Throwable {} 每个类都从 java.lang.Object 继承 finalize() 方法 当垃圾收集器确定不再存在对该对象的引用时调用该方法 Object finalize 方法不执行任何操作,但它可以被任何类覆盖,通常它应该被覆盖以清理非 Java 资源,即如果覆盖 finalize(),则关闭文件使用 try-catch-finally 语句并始终调用 super.finalize( )。这是一项安全措施,可确保您不会无意中错过关闭调用 class protected void finalize() throws Throwable { try { close(); // 关闭打开的文件 } finally { super.finalize();在垃圾回收期间由 finalize() 抛出的任何异常都会停止终结,但否则会被忽略 finalize() 永远不会在任何对象上运行超过一次

引自:http://www.janeg.ca/scjp/gc/finalize.html

你也可以看看这篇文章:

对象终结和清理


您链接到的 JavaWorld 文章来自 1998 年,并且有一些有趣的建议,特别是建议调用 System.runFinalizersOnExit() 以确保终结器在 JVM 退出之前运行。该方法目前已被弃用,并带有注释“此方法本质上是不安全的。它可能导致在其他线程同时操作这些对象时对活动对象调用终结器,从而导致行为不稳定或死锁。所以我不会那样做。
由于 runFinalizerOnExit() 不是线程安全的,因此可以做的是 Runtime.getRuntime().addShutdownHook(new Thread() { public void run() { destroyMyEnclosingClass(); } });在类的构造函数中。
@Ustaman Sangat 这是一种方法,但请记住,这会从shutdownHook 设置对您的实例的引用,这几乎可以保证您的类永远不会被垃圾收集。换句话说,这是一个内存泄漏。
@pieroxy,虽然我同意这里的其他人关于不使用 finalize() 做任何事情,但我不明白为什么必须从关闭挂钩中引用。可以有一个软参考。
r
rsp

Java finalize() 方法不是析构函数,不应用于处理应用程序所依赖的逻辑。 Java 规范声明不能保证在应用程序的生存期内完全调用 finalize 方法。

您可能想要的是 finally 和清理方法的组合,如:

MyClass myObj;

try {
    myObj = new MyClass();

    // ...
} finally {
    if (null != myObj) {
        myObj.cleanup();
    }
}

这将正确处理 MyClass() 构造函数引发异常的情况。


@Anatoly 编辑错误,它错过了 MyClass() 构造函数抛出异常的情况。这将在新版本的示例中触发 NPE。
r
rmtheis

查看 Effective Java,第 2 版第 27 页。第 7 项:避免使用终结器

终结器是不可预测的,通常很危险,而且通常是不必要的。永远不要在终结器中做任何时间紧迫的事情。永远不要依赖终结器来更新关键的持久状态。

要终止资源,请改用 try-finally:

// try-finally 块保证终止方法的执行 Foo foo = new Foo(...); try { // 用 foo 做必须做的事情 ... } finally { foo.terminate(); // 显式终止方法 }


或使用 try-with-resources
这假设对象的生命周期在一个函数的范围内。当然,这不是 OP 所指的情况,也不是任何人都需要的情况。想想“缓存引擎返回值引用计数”。您想在最后一个 ref 被释放时释放缓存条目,但您不知道最后一个 ref 何时被释放。 finalize() 可以减少引用计数,例如......但是如果您要求用户显式调用免费函数,则您要求内存泄漏。通常我两者都做(释放功能+双重检查最终确定......)......
S
Stephen C

Java 什么时候调用 finalize() 方法?

finalize 方法将在 GC 检测到对象不再可达之后,并且在它实际回收对象使用的内存之前被调用。

如果一个对象永远不会变得不可访问,finalize() 将永远不会被调用。

如果 GC 没有运行,那么 finalize() 可能永远不会被调用。 (通常,GC 仅在 JVM 确定可能有足够的垃圾使其值得时才运行。)

在 GC 确定特定对象不可达之前,可能需要一个以上的 GC 周期。 (Java GC 通常是“世代”收集器......)

一旦 GC 检测到一个对象不可访问且可终结,它就会被放置在终结队列中。终结通常与正常 GC 异步发生。

(JVM 规范实际上允许 JVM 从不运行终结器……前提是它不回收对象使用的空间。以这种方式实现的 JVM 将被削弱/无用,但这种行为是“允许的” .)

结果是,依靠最终确定来完成必须在确定的时间范围内完成的事情是不明智的。根本不使用它们是“最佳实践”。应该有一种更好(即更可靠)的方法来执行您在 finalize() 方法中尝试执行的任何操作。

最终确定的唯一合法用途是清理与应用程序代码丢失的对象相关的资源。即使这样,您也应该尝试编写应用程序代码,以便它不会首先丢失对象。 (例如,使用 Java 7+ try-with-resources 以确保始终调用 close() ...)

我创建了一个测试类,它在通过覆盖调用 finalize() 方法时写入文件。它不被执行。谁能告诉我它没有执行的原因?

很难说,但有几种可能:

该对象不会被垃圾回收,因为它仍然可以访问。

该对象不会被垃圾回收,因为在您的测试完成之前 GC 不会运行。

该对象由 GC 找到并由 GC 放入终结队列中,但在您的测试完成之前终结并未完成。


c
cst1992

由于 JVM 调用 finalize() 方法存在不确定性(不确定是否会执行被覆盖的 finalize()),为了研究目的,观察调用 finalize() 时发生的情况的更好方法是通过命令 System.gc() 强制 JVM 调用垃圾回收。

具体来说,finalize() 在对象不再使用时被调用。但是当我们试图通过创建新对象来调用它时,它的调用是不确定的。因此,为了确定,我们创建了一个 null 对象 c,它显然没有将来使用,因此我们看到对象 c 的 finalize 调用。

例子

class Car {

    int maxspeed;

    Car() {
        maxspeed = 70;
    }

    protected void finalize() {

    // Originally finalize method does nothing, but here we override finalize() saying it to print some stmt
    // Calling of finalize is uncertain. Difficult to observe so we force JVM to call it by System.gc(); GarbageCollection

        System.out.println("Called finalize method in class Car...");
    }
}

class Bike {

    int maxspeed;

    Bike() {
        maxspeed = 50;
    }

    protected void finalize() {
        System.out.println("Called finalize method in class Bike...");
    }
}

class Example {

    public static void main(String args[]) {
        Car c = new Car();
        c = null;    // if c weren`t null JVM wouldn't be certain it's cleared or not, null means has no future use or no longer in use hence clears it
        Bike b = new Bike();
        System.gc();    // should clear c, but not b
        for (b.maxspeed = 1; b.maxspeed <= 70; b.maxspeed++) {
            System.out.print("\t" + b.maxspeed);
            if (b.maxspeed > 50) {
                System.out.println("Over Speed. Pls slow down.");
            }
        }
    }
}

输出

    Called finalize method in class Car...
            1       2       3       4       5       6       7       8       9
    10      11      12      13      14      15      16      17      18      19
    20      21      22      23      24      25      26      27      28      29
    30      31      32      33      34      35      36      37      38      39
    40      41      42      43      44      45      46      47      48      49
    50      51Over Speed. Pls slow down.
            52Over Speed. Pls slow down.
            53Over Speed. Pls slow down.
            54Over Speed. Pls slow down.
            55Over Speed. Pls slow down.
            56Over Speed. Pls slow down.
            57Over Speed. Pls slow down.
            58Over Speed. Pls slow down. 
            59Over Speed. Pls slow down.
            60Over Speed. Pls slow down.
            61Over Speed. Pls slow down.
            62Over Speed. Pls slow down.
            63Over Speed. Pls slow down.
            64Over Speed. Pls slow down.
            65Over Speed. Pls slow down.
            66Over Speed. Pls slow down.
            67Over Speed. Pls slow down.
            68Over Speed. Pls slow down.
            69Over Speed. Pls slow down.
            70Over Speed. Pls slow down.

注意 - 即使在打印到 70 之后并且在程序中没有使用对象 b 之后,也不确定 b 是否被 JVM 清除,因为没有打印“在类 Bike 中调用 finalize 方法...”。


调用 System.gc(); 并不能保证垃圾收集将实际运行。
也不能保证将运行什么样的集合。相关,因为大多数 Java GC 是“世代”收集器。
O
OnaBai

finalize 将打印出类创建的计数。

protected void finalize() throws Throwable {
    System.out.println("Run F" );
    if ( checkedOut)
        System.out.println("Error: Checked out");
        System.out.println("Class Create Count: " + classCreate);
}

主要的

while ( true) {
    Book novel=new Book(true);
    //System.out.println(novel.checkedOut);
    //Runtime.getRuntime().runFinalization();
    novel.checkIn();
    new Book(true);
    //System.runFinalization();
    System.gc();

如你看到的。以下输出显示了当类数为 36 时 gc 第一次执行。

C:\javaCode\firstClass>java TerminationCondition
Run F
Error: Checked out
Class Create Count: 36
Run F
Error: Checked out
Class Create Count: 48
Run F

M
Martin Kersten

最近与终结器方法搏斗(为了在测试期间处理连接池),我不得不说终结器缺少很多东西。使用 VisualVM 观察以及使用弱引用来跟踪实际交互我发现在 Java 8 环境(Oracle JDK、Ubuntu 15)中以下情况是正确的:

Finalize 不会立即被调用,Finalizer(GC 部分)单独拥有引用难以捉摸

默认垃圾收集器池无法访问的对象

Finalize 被批量调用,指向一个实现细节,即垃圾收集器在某个阶段释放资源。

调用 System.gc() 通常不会导致对象更频繁地被终结,它只会导致终结器更快地意识到无法访问的对象

由于在执行堆转储或某些其他内部机制期间的高堆开销,创建线程转储几乎总是会导致触发终结器

终结接缝受内存要求(释放更多内存)或标记为终结增长的某个内部限制的对象列表的约束。因此,如果您有很多对象要完成,那么与只有少数几个相比,完成阶段会更频繁、更早地触发

在某些情况下,System.gc() 直接触发了 finalize,但前提是引用是本地的且短暂的。这可能与世代有关。

最后的想法

Finalize 方法不可靠,但只能用于一件事。您可以确保对象在被垃圾收集之前已关闭或处置,如果正确处理涉及生命周期结束操作的具有更复杂生命周期的对象,则可以实现故障安全。这就是我能想到的让它值得覆盖它的一个原因。


D
Darin Kolev

如果一个对象无法从任何活动线程或任何静态引用访问,则该对象有资格进行垃圾收集或 GC,换句话说,如果一个对象的所有引用都为空,则您可以说该对象有资格进行垃圾收集。循环依赖不计为引用,因此如果对象 A 具有对象 B 的引用并且对象 B 具有对象 A 的引用并且它们没有任何其他实时引用,那么对象 A 和 B 都将有资格进行垃圾收集。通常,在以下情况下,对象将有资格在 Java 中进行垃圾回收:

该对象的所有引用都显式设置为空,例如 object = null 对象在块内创建,一旦控制退出该块,引用就会超出范围。父对象设置为 null,如果一个对象持有另一个对象的引用,并且当您将容器对象的引用设置为 null 时,子对象或包含的对象自动成为垃圾回收的条件。如果一个对象只有通过 WeakHashMap 的实时引用,它将有资格进行垃圾收集。


如果在缓慢计算期间重复使用对象的 final 字段,并且此后将永远不会使用该对象,那么该对象是否会一直保持活动状态,直到源代码最后一次请求该字段,或者 JIT 是否可以复制字段到一个临时变量,然后在计算前放弃该对象?
@supercat:优化器可能会将代码重写为最初未创建对象的形式;在这种情况下,它可能会在其构造函数完成后立即完成,除非同步强制在对象的使用和终结器之间进行排序。
@Holger:如果 JIT 可以看到对象在创建和放弃之间发生的所有事情,我认为 JIT 没有任何理由提前触发终结器。真正的问题是代码需要做什么来确保终结器不能在特定方法中触发。在 .NET 中有一个 GC.KeepAlive() 函数,它除了强制 GC 假设它可能使用一个对象之外什么都不做,但我知道 Java 中没有这样的函数。可以为此目的使用 volatile 变量,但仅将变量用于此目的似乎很浪费。
@supercat:JIT 不会触发终结,它只是安排代码不保留引用,但是,直接将 FinalizerReference 排入队列可能是有益的,因此它不需要 GC 循环来找出没有参考资料。同步足以确保 happens-before 关系;由于最终确定可能(实际上是)在不同的线程中运行,因此无论如何它通常是正式必要的。 Java 9 将添加 Reference.reachabilityFence
@Holger:如果 JIT 优化了对象创建,那么唯一会调用 Finalize 的就是 JIT 生成的代码直接这样做了。对于只希望在单个线程中使用的对象,通常需要同步代码吗?如果一个对象在它被放弃之前执行了一些需要撤消的操作(例如打开一个套接字连接并获得对另一端资源的独占使用),让终结器在代码仍在使用套接字时关闭连接将是一场灾难.代码使用同步是否正常...
C
Craig S. Anderson

不保证 finalize 方法。当对象符合 GC 条件时调用此方法。有很多情况下对象可能不会被垃圾回收。


不正确。你的意思是当一个对象变得无法访问时,它就被最终确定了。它实际上是在实际收集方法时调用的。
A
Amarildo

有时当它被销毁时,一个对象必须做出一个动作。例如,如果对象具有文件句柄或字体等非 java 资源,则可以在销毁对象之前验证这些资源是否已释放。为了管理这种情况,java 提供了一种称为“finalizing”的机制。通过完成它,您可以定义当一个对象即将从垃圾收集器中删除时发生的特定操作。要将终结器添加到类中,只需定义 finalize() 方法。每当 Java 执行时将要删除该类的对象时,都会调用此方法。在 finalize method() 中,您指定在销毁对象之前要执行的操作。垃圾收集器会定期搜索不再引用任何运行状态或间接引用任何其他对象的对象。在资产被释放之前,Java 运行时调用对象的 finalize() 方法。 finalize() 方法具有以下一般形式:

protected void finalize(){
    // This is where the finalization code is entered
}

使用 protected 关键字,可以防止类外的代码访问 finalize()。重要的是要理解 finalize() 是在垃圾回收之前调用的。例如,当对象离开范围时,它不会被调用。这意味着您不知道何时或是否会执行 finalize()。因此,程序必须提供其他方法来释放系统资源或对象使用的其他资源。您不应该依赖 finalize() 来正常运行程序。


M
Meet Doshi

我们覆盖 finalize 方法的类

public class TestClass {    
    public TestClass() {
        System.out.println("constructor");
    }

    public void display() {
        System.out.println("display");
    }
    @Override
    public void finalize() {
        System.out.println("destructor");
    }
}

调用 finalize 方法的机会

public class TestGarbageCollection {
    public static void main(String[] args) {
        while (true) {
            TestClass s = new TestClass();
            s.display();
            System.gc();
        }
    }
}

当内存被转储对象超载时,gc 将调用 finalize 方法

运行并查看控制台,您不会发现经常调用 finalize 方法,当内存过载时,将调用 finalize 方法。


F
Fourat

finalize() 在垃圾回收之前调用。当对象超出范围时不会调用它。这意味着您无法知道何时或什至是否会执行 finalize()

例子:

如果您的程序在垃圾收集器发生之前结束,则 finalize() 将不会执行。因此,它应该用作备份程序以确保正确处理其他资源,或用于特殊用途的应用程序,而不是作为您的程序在其正常运行中使用的手段。


J
JavaDev

Java 允许对象实现一个名为 finalize() 的方法,该方法可能会被调用。如果垃圾收集器尝试收集对象,则调用 finalize() 方法。如果垃圾收集器没有运行,则不会调用该方法。如果垃圾收集器未能收集对象并尝试再次运行它,则不会在第二次调用该方法。实际上,您不太可能在实际项目中使用它。请记住,它可能不会被调用,并且绝对不会被调用两次。 finalize() 方法可以运行零次或一次。在下面的代码中,finalize() 方法在我们运行它时不会产生任何输出,因为程序在需要运行垃圾收集器之前就退出了。

Source


U
Utkarsh

正如 https://wiki.sei.cmu.edu/confluence/display/java/MET12-J.+Do+not+use+finalizers 中指出的,

由于执行时间取决于 Java 虚拟机 (JVM),因此没有必须执行终结器的固定时间。唯一的保证是,任何执行的终结器方法都会在关联对象变得无法访问(在垃圾回收的第一个周期期间检测到)之后和垃圾回收器回收关联对象的存储之前(在垃圾回收器的第二个周期期间)执行此操作.在对象变得无法访问后,对象的终结器的执行可能会延迟任意长的时间。因此,在对象的 finalize() 方法中调用时间关键功能(例如关闭文件句柄)是有问题的。


P
Panagiotis Bougioukos

来自 JDK 18 的最新消息

根据 openjdk 18 上交付的 JEPS 421finalize() 方法的最终确定和因此的功能将被标记为 deprecated(forRemoval=true),这意味着永久删除将在 jdk 18 之后的某些更高版本中进行。

从 jdk 18 开始,一个新的命令行选项 --finalization=disabled 会在任何地方禁用终结机制,即使对于 jdk 本身内部的声明也是如此。

这也与这里的这个问题有关,因为它计划删除的原因是它包含的一些主要缺陷。其中一个缺陷是,从对象变得无法访问到调用其终结器的那一刻可能会经过很长时间。 GC 也不保证会调用任何终结器,这也是事实。


F
Fruchtzwerg

尝试运行此程序以更好地理解

public class FinalizeTest 
{       
    static {
        System.out.println(Runtime.getRuntime().freeMemory());
    }

    public void run() {
        System.out.println("run");
        System.out.println(Runtime.getRuntime().freeMemory());
    }

     protected void finalize() throws Throwable { 
         System.out.println("finalize");
         while(true)
             break;          
     }

     public static void main(String[] args) {
            for (int i = 0 ; i < 500000 ; i++ ) {
                    new FinalizeTest().run();
            }
     }
}