【Java面试】说说你对finalize的理解

简介: 【Java面试】说说你对finalize的理解

面试官:说说你对finalize方法的理解

答:它是Object中的一个方法,子类重写他,垃圾回收时此方法将会被调用,可以在其中进行一些资源的释放和清理工作,但是将资源释放和清理放在finalize方法中非常不好,非常影响性能,严重的时候甚至会引起OOM,从Java9开始就被被标注为@Deprecated,不建议被使用了。

面试官:为什么?

答:当重写了finalize方法的对象,在类的构造方法调用之时,JVM都会将其包装成一个Finalizer对象,并加入unfinalized 队列中(静态成员变量、双向链表结构)。同时这些类也会指向一个ReferenceQueue类型,它也是 Finalizer 类中一个静态成员变量,名为queue(是一个单向链表结构),刚开始它是空的。当对象可以被当作垃圾回收时,就会把这些对象对应的Finalizer对象加入这个队列。即使这些对象没人引用,垃圾回收的时候也无法立即被回收,因为要使得重写了finalize方法的类被回收的时候能够调用finalize方法(因为如果先被回收了,那么就没办法调用finalize方法了),那么此时就需要一个FinalizerThread线程,它会去从ReferenceQueue中逐一取出每个Finalizer对象,并把他们从链表断开,这样就没有引用能引用到他了,那么此时下次gc的时候这个对象就可以被回收了。而回收这个对象是通过Finalizer这个守护线程进行回收的。

并且如果在调用finalize方法的时候出现了异常,是不会报错的,这就会导致后期如果代码出错了,很难排查。

答:因此我做一下总结:重写finalize有两个非常不好的点。

  • FinalizerThread是守护线程,代码很有可能没来得及执行完,线程就结束了,造成资源没有正确释放
  • 异常被吞掉这个就太糟了,甚至不能判断有没有在释放资源时发生错误

其次就是影响性能,重写了 finalize方法的对象在第一次被gc时,并不能及时释放它占用的内存,因为要等着FinalizerThread调用完finalize,把它从第一个unfinalized 队列移除后,第二次gc时才能真正释放内存。

同时可以想象gc本就因为内存不足引起,finalize调用又很慢(两个队列的移除操作,都是串行执行的,用来释放连接类的资源也应该不快),不能及时释放内存,对象释放不及时就会逐渐移入老年代,老年代垃圾积累过多就会容易full gc,full gc后释放速度如果仍跟不上创建新对象的速度,就会OOM。

当然,有一些文章提到【Finalizer线程会和我们的主线程进行竞争,不过由于它的优先级较低,获取到的CPU时间较少,因此它永远也赶不上主线程的步伐】这个显然是错误的,FinalizerThread的优先级较普通线程更高,赶不上步伐的原因应该是finalize 执行慢等原因综合导致的。

面试官:6


目录
打赏
0
0
0
0
5
分享
相关文章
Java社招面试中的高频考点:Callable、Future与FutureTask详解
大家好,我是小米。本文主要讲解Java多线程编程中的三个重要概念:Callable、Future和FutureTask。它们在实际开发中帮助我们更灵活、高效地处理多线程任务,尤其适合社招面试场景。通过 Callable 可以定义有返回值且可能抛出异常的任务;Future 用于获取任务结果并提供取消和检查状态的功能;FutureTask 则结合了两者的优势,既可执行任务又可获取结果。掌握这些知识不仅能提升你的编程能力,还能让你在面试中脱颖而出。文中结合实例详细介绍了这三个概念的使用方法及其区别与联系。希望对大家有所帮助!
191 60
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
90 14
Java 面试必问!线程构造方法和静态块的执行线程到底是谁?
大家好,我是小米。今天聊聊Java多线程面试题:线程类的构造方法和静态块是由哪个线程调用的?构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节有助于掌握Java多线程机制。下期再见! 简介: 本文通过一个常见的Java多线程面试题,详细讲解了线程类的构造方法和静态块是由哪个线程调用的。构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节对掌握Java多线程编程至关重要。
54 13
Java线程调度揭秘:从算法到策略,让你面试稳赢!
在社招面试中,关于线程调度和同步的相关问题常常让人感到棘手。今天,我们将深入解析Java中的线程调度算法、调度策略,探讨线程调度器、时间分片的工作原理,并带你了解常见的线程同步方法。让我们一起破解这些面试难题,提升你的Java并发编程技能!
98 16
Java 高级面试技巧:yield() 与 sleep() 方法的使用场景和区别
本文详细解析了 Java 中 `Thread` 类的 `yield()` 和 `sleep()` 方法,解释了它们的作用、区别及为什么是静态方法。`yield()` 让当前线程释放 CPU 时间片,给其他同等优先级线程运行机会,但不保证暂停;`sleep()` 则让线程进入休眠状态,指定时间后继续执行。两者都是静态方法,因为它们影响线程调度机制而非单一线程行为。这些知识点在面试中常被提及,掌握它们有助于更好地应对多线程编程问题。
80 9
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
75 12
Java Dubbo 面试题
Java Dubbo相关基础面试题
Java MyBatis 面试题
Java MyBatis相关基础面试题
Java JVM 面试题
Java JVM(虚拟机)相关基础面试题
Java Druid 面试题
Java Druid 连接池相关基础面试题

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等