如何在不使用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-...
你 发表评论:
欢迎- 一周热门
-
-
UOS服务器操作系统防火墙设置(uos20关闭防火墙)
-
极空间如何无损移机,新Z4 Pro又有哪些升级?极空间Z4 Pro深度体验
-
NAS:DS video/DS file/DS photo等群晖移动端APP远程访问的教程
-
如何在安装前及安装后修改黑群晖的Mac地址和Sn系列号
-
手机如何设置与显示准确时间的详细指南
-
如何修复用户配置文件服务在 WINDOWS 上登录失败的问题
-
一加手机与电脑互传文件的便捷方法FileDash
-
日本海上自卫队的军衔制度(日本海上自卫队的军衔制度是什么)
-
10个免费文件中转服务站,分享文件简单方便,你知道几个?
-
爱折腾的特斯拉车主必看!手把手教你TESLAMATE的备份和恢复
-
- 最近发表
- 标签列表
-
- linux 查询端口号 (58)
- docker映射容器目录到宿主机 (66)
- 杀端口 (60)
- yum更换阿里源 (62)
- internet explorer 增强的安全配置已启用 (65)
- linux自动挂载 (56)
- 禁用selinux (55)
- sysv-rc-conf (69)
- ubuntu防火墙状态查看 (64)
- windows server 2022激活密钥 (56)
- 无法与服务器建立安全连接是什么意思 (74)
- 443/80端口被占用怎么解决 (56)
- ping无法访问目标主机怎么解决 (58)
- fdatasync (59)
- 405 not allowed (56)
- 免备案虚拟主机zxhost (55)
- linux根据pid查看进程 (60)
- dhcp工具 (62)
- mysql 1045 (57)
- 宝塔远程工具 (56)
- ssh服务器拒绝了密码 请再试一次 (56)
- ubuntu卸载docker (56)
- linux查看nginx状态 (63)
- tomcat 乱码 (76)
- 2008r2激活序列号 (65)