百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

如何在不使用finalize的情况下处理Java的错误和资源清理(翻译)

nanshan 2024-11-10 10:10 17 浏览 0 评论


Java的finalize()方法将在Java 18中不建议使用,并在未来的版本中完全删除。我们来看看其他选择。


经过几年的隆隆声,Java正准备弃用JDK 18中的finalize()方法。这包含在JDK增强提案421中,该提案将标记为已弃用,并允许将其关闭进行测试。它将保持默认启用状态。它将在未来的版本中完全删除。借此机会,让我们看看finalize ()的结束意味着什么,以及我们现在应该如何处理错误和资源清理。


什么是finalize?

在我们理解为什么finalize会消失以及使用什么之前,让我们先了解什么是finalize

基本想法是允许您在对象上定义一种方法,该方法将在对象准备进行垃圾收集时执行。从技术上讲,当对象变得可以访问幽灵时,它已准备好进行垃圾收集,这意味着JVM中没有留下强引用或弱引用。

此时,想法是JVM将执行object.finalize()方法,然后特定于应用程序的代码将清理任何资源,例如I/O流或数据存储的句柄。

Java中的根Object类有一个finalize()方法,以及其他方法,如equals()和hashCode()。这是为了使每个Java程序中的每个对象都能参与这个简单的机制,以避免资源泄漏。

请注意,这也解决了抛出异常和可能遗漏其他清理代码的情况:该对象仍将被标记为垃圾收集,最终将调用其最终方法。问题解决了,对吗?只需在耗费资源的对象上覆盖finalize()方法。

使用finalize的问题

这就是想法,但现实完全是另一回事。finalize存在一些缺陷,阻碍了清理utopia的实现。(这种情况类似于serialize(),另一种在纸面上看起来不错,但在实践中变得有问题的方法。)

finalize的问题包括:

Finalize可能会以意想不到的方式运行。有时,GC会在您认为之前确定您的对象没有对它的实时引用。看看这个相当可怕的堆栈溢出答案。

Finalize可能永远不会运行,或者会在长时间延迟后运行。在频谱的另一端,您的finalize可能永远不会运行。正如JEP 421 RFC所说,“GC通常仅在必要时运行以满足内存分配请求。”所以你任由GC突发奇想摆布。

Finalize 可以使原本死掉的类复活。有时,一个对象会触发一个异常,使其符合 GC 条件。然而,finalize() 方法有机会首先运行,并且该方法可以做任何事情,包括重新建立对对象的实时引用。这是一个潜在的泄漏源和安全隐患。

Finalize 很难正确实现。直接编写一个功能强大且无错误的可靠 finalize 方法并不像看起来那么容易。特别是,不能保证 finalize 的线程含义。finalizers可以在任何线程上运行,从而引入非常难以调试的错误条件。忘记调用 finalize() 会导致难以发现的问题。

性能问题。鉴于 finalize 在实现其既定目的时的不可靠性,JVM 支持它的开销是不值得的。

Finalize 使得更脆弱的大规模应用程序变得更加脆弱。研究得出的结论是,使用 finalize 的大型软件更容易脆弱,并且会遇到在重负载下出现的难以重现的错误情况。

finalize之后的生命周期

Finalize 使得更脆弱的大规模应用程序变得更加脆弱。研究得出的结论是,使用 finalize 的大型软件更容易脆弱,并且会遇到在重负载下出现的难以重现的错误情况。

Try-catch-finally 语句块

处理资源释放的老式方法是通过 try-catch 块。这在许多情况下都是可行的,但它容易出错且冗长。例如,要完全捕获嵌套错误条件(即关闭资源时也会引发异常),您需要类似于清单 1 的内容。

这似乎有点矫枉过正,但在一个长期运行且使用量很大的系统中,这些情况可能会导致资源泄漏,从而导致应用程序崩溃。因此,唉,必须在整个代码库中重复冗长。这些东西因破坏代码流而臭名昭著。

FileOutputStream outStream = null;
try {
  outStream = new FileOutputStream("output.file");
  ObjectOutputStream stream = new ObjectOutputStream(outStream);
  stream.write //…
  stream.close();
} catch(FileNotFoundException ffe) {
  throw new RuntimeException("Could not open file for writing", ffee);
} catch(IOException ioe) {
  System.err.println("Error writing to file");
} finally {
  if (outStream != null) {
    try {
      outStream.close();
    } catch (Exception e) {
      System.err.println(“Failed to close stream”, e);
    }
  }
}


在清单 1 中您想要做的就是打开一个流,向它写入一些字节,并确保它被关闭,而不管抛出什么异常。为此,您必须将调用包装在 try 块中,并且如果引发任何已检查的异常,请处理它们(通过引发包装的运行时异常或将异常打印到日志中)。

然后,您需要添加一个 finally 块来仔细检查流。这是为了确保异常不会阻止关闭。但是你不能只是关闭流;您必须将其包装在另一个 try 块中,以确保关闭不会自行出错。对于一个简单而常见的需求,这需要大量的工作和中断。

Try-with-resource 语句

在 Java 7 中引入的 try-with-resource 语句允许您指定一个或多个资源对象作为 try 声明的一部分。当 try 块完成时,这些资源保证会被关闭。

具体来说,任何实现 java.lang.AutoCloseable 的类都可以提供给 try-with-resource。这几乎涵盖了您在 Java 生态系统中可以找到的所有常用资源。

让我们重写清单 1 以使用 try-with-resource 语句,如清单 2 所示。

try (FileOutputStream outStream = new ObjectOutputStream(outStream)) {
  ObjectOutputStream stream = new ObjectOutputStream(outStream);
  stream.write //…
  stream.close();
} catch(FileNotFoundException ffe) {
  throw new RuntimeException("Could not open file for writing", ffee);
} catch(IOException ioe) {
  System.err.println("Error writing to file");
}


您可以看到这里有许多好处可以减少代码占用,但最大的好处是,一旦您通过在 try 块括号内声明流(或您正在使用的任何内容)将其移交给 VM,您不必再担心了。它将为您关闭。没有资源泄漏。我们已经消除了对 finally 块或任何最终确定的调用的需要。这解决了大多数用例的主要问题(尽管检查错误处理的冗长仍然存在)。

在某些情况下,当事情太复杂而无法像这样在单个块中处理时,需要更精细和更强大的解决方案。对于这些情况,Java 开发人员需要更强大的东西。对于这些情况,您需要清洁工。

相关推荐

Linux 的磁盘系统,和你了解的Windows差别很大

我的C盘去哪了?一个系统,如果没有存储,那么也就不能称之为系统。存储性是一个完整系统的重要组成部分。例如AWS最开始的服务就是S3(用来存储数据的云服务),足以见得存储对于一个应用平台是多么的重要。...

一文读懂 Linux 硬盘挂载:从问题到解决方案

各位互联网大厂的后端开发伙伴们!在咱们日常工作中,操作Linux系统是常有的事儿吧。你们有没有遇到过这样的场景:新添加了一块硬盘,满心欢喜准备用来存储重要数据或者部署新的应用服务,却突然发现不知道...

硬盘分区(硬盘分区格式)

 磁盘(硬盘)分区,可以分C、D、E等分区,大家可能都会用,会根据自已的需要确定所需的空间,但分区是如何工作的呢,内容如下。Windows中有3类:MBR分区:MasterBootRecord,也...

parted命令工具分区介绍(particle命令)

linux系统磁盘分区通常可以使用fdisk和parted命令,当分区大小小于2TB的时候,两种皆可以使用,当分区大于2TB的话,就需要用parted分区。以下介绍parted命令相关使用,以sdb为...

Linux 服务器上查看磁盘类型的方法

方法1:使用lsblk命令lsblk输出说明:TYPE列显示设备类型,如disk(物理磁盘)、part(分区)、rom(只读存储)等。NAME列显示设备名称(如sda、nvme0n1)。TR...

Linux分区命令fdisk和parted使用介绍

摘要:一般情况下,Linux分区都是选择fdisk工具,要求硬盘格式为MBR格式,能支持的最大分区空间为2T。但是目前在实际生产环境中使用的磁盘空间越来越大,呈TB级别增长;而常用的fdisk这个工具...

linux 分区原理与名词解释(linux操作系统中的分区类型)

分区的意义将磁盘分成几份,每份挂在到文件系统的那个目录在linux里的文件系统Ext2:早期的格式,不支持日志功能Ext3:ext2改良版,增加了日志功能,是最基本且最常用的使用格式了Ext4:针对e...

linux 分区合并(linux合理分区)

查看虚拟机当前磁盘挂载情况fdisk-l选择磁盘fdisk/dev/sda查看磁盘分区情况p重新选择分区n选择主分区p保存w创建物理卷pvcreate/dev/sda3查看物理卷信息pvdi...

如何在 Linux 系统中永久禁用交换分区 ?

Linux操作系统中的交换分区或交换文件充当硬盘上的临时存储区域,当物理内存(RAM)满时,系统使用该存储区域。它用于交换较少使用的内存页,这样系统就不会因为运行应用程序而耗尽物理内存。随着技术的发...

Linux 如何知道硬盘已用多少空间、未用多少空间

刚出社会时,去了一家公司上班,老板为了省钱,买的服务器是低配的,硬盘大小只有40G,有一次网站突然不能访问了,排查半天才知道原来服务器的硬盘空间已用完,已无可用空间。第一步是查看硬盘的使用情况,第二步...

用Linux系统管理磁盘空间 就该这么来

要想充分有效的管理使用Linux系统中的存储空间,用户必须要做的就是双管齐下,一边扩充空间一边限制空间。不得不说的就是很多时候磁盘空间就像水资源,需节制水流。说到要如何实现限制空间就离不开使用LVM技...

Windows 11 磁盘怎么分区?(windows11磁盘怎么分区)

Windows11磁盘分区技术解析与操作指南:构建高效存储体系一、磁盘分区的技术本质与系统价值磁盘分区作为存储系统的基础架构,通过逻辑划分实现数据隔离与管理优化。Windows11采用NTF...

linux上创建多个文件分区,格式化为 ext2、ext3、ext4、XFS 文件

以下是在Linux系统上创建多个20GB文件分区并格式化为不同文件系统的分步指南:步骤1:创建基础文件(4个20GB文件)bash#创建4个20GB稀疏文件(实际占用空间随写入量增长)ddif=/...

救命的U盘低格哪家最强?(低格优盘)

周二时有位童鞋留言说U盘之前做过引导盘,现在格式化不了,用各种工具都不行,而且因为U盘厂商的关系,查不到U盘主控,无法量产恢复,特来求助。小编花了点时间特意弄坏一个U盘分区,终于试出方法了,特来分享一...

Linux 查看硬件磁盘存储大小和磁盘阵列(RAID)的组合方式

一、查看硬件磁盘存储大小查看所有磁盘信息:#lsblk该命令会列出所有磁盘(如/dev/sda、/dev/nvme0n1)及其分区和挂载点。查看磁盘总容量:fdisk-l#或parted-...

取消回复欢迎 发表评论: