Java中的RAII

资源获取即初始化( RAII )是Bjarne Stroustrup用C ++引入的一种用于异常安全资源管理的设计思想。 感谢垃圾回收,Java 没有此功能,但是我们可以使用try-with-resources实现类似的功能。

约翰·哈德斯(John Huddles)在萨赫姆农场(1998)

RAII解决的问题很明显; 看一下这段代码(我确定您知道Semaphore是什么以及它在Java中的工作方式):

class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {this.sem.acquire();if (x > 1000) {throw new Exception("Too large!");}System.out.printf("x = %d", x);this.sem.release();}
}

该代码相当原始,没有任何用处,但您很可能会明白:如果从多个并行线程调用方法print() ,则只允许其中五个并行打印。 有时,如果x大于1000 ,将不允许其中的一些打印,并且将引发异常。

该代码的问题是资源泄漏 。 x大于1000每个print()调用都会从信号量中获取一个许可,并且不会返回它。 在五个异常调用中,信号量将为空,所有其他线程将不打印任何内容。

解决办法是什么? 这里是:

class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {this.sem.acquire();if (x > 1000) {this.sem.release();throw new Exception("Too large!");}System.out.printf("x = %d", x);this.sem.release();}
}

抛出异常之前,我们必须释放许可证。

但是,还有另一个问题出现:代码重复。 我们在两个地方发放许可证。 如果我们添加更多的throw指令,我们还必须添加更多的sem.release()调用。

在C ++中引入了一个非常优雅的解决方案,称为RAII。 这是在Java中的外观:

class Permit {private Semaphore sem;Permit(Semaphore s) {this.sem = s;this.sem.acquire();}@Overridepublic void finalize() {this.sem.release();}
}
class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {new Permit(this.sem);if (x > 1000) {throw new Exception("Too large!");}System.out.printf("x = %d", x);}
}

看看方法Foo.print()的代码多么漂亮。 我们只创建了Permit类的实例,它立即在信号量上获得了一个新的许可。 然后我们通过异常或正常方式退出方法print() ,然后方法Permit.finalize()释放许可。

优雅,不是吗? 是的,是的,但是在Java中不起作用。

它不起作用,因为与C ++不同,Java在可见性关闭时不会破坏对象。 当我们退出方法print()时, Permit类的对象不会被破坏。 它最终将被销毁,但我们不知道确切的时间。 获得信号灯中的所有许可之后,很可能将其破坏,并且我们将被阻止。

Java中也有一个解决方案。 它不像C ++那样优雅,但确实有效。 这里是:

class Permit implements Closeable {private Semaphore sem;Permit(Semaphore s) {this.sem = s;}@Overridepublic void close() {this.sem.release();}public Permit acquire() {this.sem.acquire();return this;}
}
class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {try (Permit p = new Permit(this.sem).acquire()) {if (x > 1000) {throw new Exception("Too large!");}System.out.printf("x = %d", x);}}
}

请注意try块和Permit类现在实现的Closeable接口。 try块退出时,对象p将被“关闭”。 它可以在结尾处退出,也可以通过returnthrow语句退出。 无论哪种情况,都会调用Permit.close() :这就是try-with-resources在Java中的工作方式。

我介绍的方法acquire()和感动sem.acquire()出的Permit构造,因为我相信这构造函数必须是免费的代码。

总而言之,RAII是一个完美的设计 图案 处理可能泄漏的资源时的方法。 即使Java没有开箱即用,我们也可以通过try-with-resources和Closeable实现它。

翻译自: https://www.javacodegeeks.com/2017/08/raii-in-java.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/349358.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

eclipse juno_Eclipse Juno上带有GlassFish的JavaEE 7

eclipse junoJava EE 7很热。 前四个JSR最近通过了最终批准选票,与此同时GlassFish 4达到了升级版83。 如果您关注我的博客,那么您将了解NetBeans的大部分工作。 但是我确实认识到,那里还有其他IDE用户,他们也有权试用最新和最出色…

java 生成校验验证码_java 验证码生成与校验

java绘图相关类验证码工具类package dt2008.util;import javax.imageio.ImageIO;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import java.awt.*;import java.awt.image.BufferedImage;import java.io.IOException;import ja…

红黑树中nil结点_什么是红黑树?程序员面试必问!

点击上方java小组,选择“置顶公众号”优质文章,第一时间送达当在10亿数据中只需要进行10几次比较就能查找到目标时,不禁感叹编程之魅力!人类之伟大呀! —— 学红黑树有感。终于,在学习了几天的红黑树相关的…

杰克逊JSON解析错误-UnrecognizedPropertyException:无法识别的字段,未标记为可忽略[已解决]...

在解析从我们的一个RESTful Web服务接收到的JSON字符串时,我收到此错误“线程“ main”中的异常com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException:无法识别的字段“人”(类Hello $ Person),不是标记…

mysql2008数据库配置_SQL Server 2008 R2 超详细安装图文教程

这篇文章主要介绍了SQL Server 2008 R2 超详细安装图文教程,需要的朋友可以参考下一、下载SQL Server 2008 R2安装文件二、将安装文件刻录成光盘或者用虚拟光驱加载,或者直接解压,打开安装文件,出现下面的界面安装SQL Server 2008 R2需要.NET…

hdfs读写流程_深度探索Hadoop分布式文件系统(HDFS)数据读取流程

一、开篇Hadoop分布式文件系统(HDFS)是Hadoop大数据生态最底层的数据存储设施。因其具备了海量数据分布式存储能力,针对不同批处理业务的大吞吐数据计算承载力,使其综合复杂度要远远高于其他数据存储系统。因此对Hadoop分布式文件系统(HDFS)的深入研究&a…

mysql隔离级别验证_MySQL事务隔离级别以及验证

查询初始数据开启A事务  并做更新操作再另一端 B开始另一个事务查询 事务级别设置为读未提查询到事务未提交的数据 a的count修改为3 但是没有提交2 第二个级别 读已提交 避免脏读问题 但是有不可重复读问题回滚数据 修改隔离级别 确保都是 读已提交级别客户端A客户端B验证 脏…

注意力机制可视化_目标跟踪中的(STAM)时空注意力机制

目标跟踪分为单目标跟踪和多目标跟踪,单目标跟踪较为简单,这里我们只讨论多目标跟踪。多目标跟踪的遮挡问题多目标跟踪时特别容易发生目标间的相互遮挡,从而导致严重的预测偏移问题,如下图所示:红色框的行人在和蓝色框…

js给标签添加属性和值_jquery节点属性

一.节点操作1.DOM内容节点操作:​ ①innerHTML属性:设置或获取文本的内容(普通文本和标签)。​ ②innerText属性:设置或获取文本的内容(普通文本),存在兼容性问题。2.jQuery内容节点…

sci translate好用吗_228个学科分类对应12000+本SCI和SSCI期刊,总有你要的那款!

最近有很多小伙伴询问选刊的问题,而且都是非常具体的学科方向,我们的小编虽然非常热心且礼貌的回答“近期安排”,但其实我们也感觉到鸭梨山大:根据WOS最新一期(2020/9/21)名单公布,WOS目前总共收录了12266本…

arrays.sort(._Arrays.sort与Arrays.parallelSort

arrays.sort(.我们都使用Arrays.sort对对象和原始数组进行排序。 此API在下面使用合并排序或Tim排序对内容进行排序,如下所示: public static void sort(Object[] a) {if (LegacyMergeSort.userRequested)legacyMergeSort(a);elseComparableTimSort.sor…

适用于Idea的面向现代TDD的Java 8 JUnit测试模板(带有Mockito和AssertJ)

使用类似BDD的语法,Java 8和Mockito-AssertJ二重奏为Idea调整JUnit测试类模板。 本文涵盖的主题似乎很简单。 但是,根据我的培训师经验,我知道(不幸的是)这不是常见的做法。 因此,我决定写这篇简短的博客文…

Java编程字符逆序输出_用JAVA编写一程序:从键盘输入多个字符串到程序中,并将它们按逆序输出在屏幕上。...

展开全部代码如下:import java.util.Scanner;public class ScannerDemo{public static void main(String[] args) throws Exception{Scanner scannew Scanner(System.in);System.out.println("请输入内容:");String strscan.nextLine();char[]…

julia与python对比_有人说Julia比Python好,还给出了5个理由

选自medium作者:Emmett Boudreau机器之心编译参与:杜伟、张倩、肖清本文作者从速度、通用性、多重派发、适用于 ML 的程度和包管理器 5 个方面阐述了 Julia 语言相较于 Python 的优势所在。Julia 是一种多范式的函数式编程语言,用于机器学习和…

linkedhashmap获取第n个元素_机试真题分享——交换链表前后第K个元素

题目描述给定一个编码链表和一个加密条件K,对编码进行加密。加密规则:把编码从前往后开始数第K个元素和从后往前数第K个元素进行交换。注意:编码的长度为0.第一个编码的序号是1.示例:输入:[1 2 3 4 5 6] 2输出&#xf…

Spring Boot Admin –用于管理Spring Boot应用程序的Admin UI

作为微服务开发的一部分,我们许多人都将Spring Boot与Spring Cloud功能一起使用。 在微服务领域,我们将有许多Spring Boot应用程序将在相同/不同的主机上运行。 如果将Spring Actuator添加到Spring Boot应用程序中,我们将获得很多现成的端点来…

md5与des算法有何不同_Python算法详解:为什么说算法是程序的灵魂?

算法是程序的灵魂,只有掌握了算法,才能轻松地驾驭程序开发。软件开发工作不是按部就班,而是选择一种最合理的算法去实现项目功能。算法能够引导开发者在面对一个项目功能时用什么思路去实现,有了这个思路后,编程工作只需要遭循这个思路去实现即可。本章将详细讲解计算机算法的基…

高可用性(HA),会话复制,多VM Payara群集

抽象 在研究如何创建高可用性(HA)时,我发现了会话复制的多机Payara / GlassFish群集,无法在一个参考中找到所需的一切。 我认为这将是一个普遍的需求并且很容易找到。 不幸的是,我的假设是错误的。 因此,本…

如何快速弄懂一个新模型_如何评估创业项目是否靠谱?一个新的模型 | 创创锦囊...

要判断一个创业项目是否靠谱,是否能拥有广阔的市场和巨大的增长潜力,不仅是投资人关心的话题,更是每一个创业者在创业过程中不断思考的问题。投资人关注大趋势、大机会,遵循自上而下的思维模型,在心仪的赛道上寻找合适…

java编译找不到符号 int age=in.nexint()_Java报错找不到符号,小白自学求大佬解决...

import java.util.*;public class guess_1{public static void main(String[] args){Scanner innew Scanner(System.in);System.out.println("--------猜拳游戏--------");System.out.println("请出拳(1.剪刀 2.石头 3.布)");int personin.nextInt();int c…