try-with-resources自动执行释放资源

看到了try这个关键字立马就应该能想到异常处理机制try-catch-finally语句块。这里要说的东西和异常处理背后的机制其实几乎是一样的,只不过try-with-resources侧重点异常后自动释放资源,不需要我们手动去在调用close()方法。后面会从JLS层面,以及从字节码层面举例详细说明。

做什么用的

try-with-resources语句是用变量(称为资源)来参数化的,这些变量在try块执行之前被初始化,并在try块执行后以初始化的相反顺序自动关闭。当资源自动关闭时,catch子句和finally子句通常是不必要的。

一句话就是用来自动调用关闭方法,释放资源。

Java语言规范(JLS)是如何定义的

  • 在资源规范中声明的变量类型必须是AutoCloseable的子类型,否则会发生编译时错误
  • 多个资源使用';'号分割,也就说try语句块中可以写多个资源初始化。
  • 不能有重名的变量名,否则编译报错
  • 在资源规范中声明的变量如果没有显式声明为final,则隐式声明为final

有哪些使用姿势

  1. 基本使用方法,只需try(){}语句块即可
    public class FileWriter implements AutoCloseable{
    @Override
    public void close() {
    System.out.println("file writer close");
    } public void writeFile(String s) {
    System.out.println("file writer write" + s);
    }
    }
    //这里举例一个类实现了AutoCloseable接口,然后,我们就可以直接使用try资源语句块来包装它,
    //使用完不需要去手动去close(),会自动调用close方法。
    try (
    FileWriter writer = new FileWriter();
    ) {
    writer.writeFile("target.cs");
    }
    //多个子句也是一样,关闭的顺序是反着来。如下会先调用reader.close(),然后是writer.close()
    try (
    FileWriter writer = new FileWriter();
    FileReader reader = new FileReader();
    ) { writer.writeFile("target.cs");
    reader.readFile("target.cs");
    }
  2. 可以结合catch,finally像异常处理一样。嵌套什么的,也没问题
    try (FileWriter writer = new FileWriter()) {
    writer.writeFile("target.cs");
    try(FileReader reader = new FileReader()) {
    reader.readFile("target.cs");
    }
    }catch (Exception e) {
    //do exception
    }

从字节码层面详细说明JVM是如何解析和执行的

我们没有主动调用close()方法,那是何时谁调用的close呢,我们从编译后的字节码中就可以看到

假设源码如下:非常的简单,FileWriter这个类没有贴出,就是像上面的一样实现了AutoCloseable接口,然后自定义了一个writeFile方法。

public class AutoClose {
public static void main(String[] args) {
try (FileWriter writer = new FileWriter()) {
writer.writeFile("target.cs");
}
}
}

编译后字节码展示如下:

重点说明:

我这里按照字节码序号,也就是自然顺序号(也就是第一列)[第二列是PC号]

  • 可以看到第15行出现了一句,invokevirtual 字节码指令,调用了FileWriter.close()方法。
  • 另外第23行也出现了一模一样的字节码指令。这是为了处理即使异常情况下,close方法也要正常调用。
  • 如下是异常表:

    也就是说,即使调用writeFile方法异常了,也要自动调用关闭方法,所以可以看到异常表PC号是814。异常表的第二行,解释大白话就是如果close方法异常了,那就把异常加入到异常列表中,因此看到的PC号是2226。最终是将异常抛出。
  • 粉色的+19、+9那些表示的是字节码偏移量 (当前PC号加括号中的)

多个资源列表的时候等于是多个catch一样,只要初始化成功的,无论执行业务方法是否异常,都会调用它的close方法。

try-with-resources优先于try-finally

这个原则在Effective Java中也有说明。尤其是当语句块中有多个资源需要处理关闭操作的时候,嵌套try-finally语句,很可能会导致异常被抹除。而使用try-with-resources即使多个异常被禁止时,仍然可以使用getSuppresed方法访问到它们

自动调用关闭释放资源try-with-resources的更多相关文章

  1. 根据判断PC浏览器类型和手机屏幕像素自动调用不同CSS的代码

    1.媒体查询方法在 css 里面这样写 -------------------- @media screen and (min-width: 320px) and (max-width: 480px) ...

  2. paintEvent(QPaintEvent*)是系统自动调用的

    qt中函数paintEvent(QPaintEvent*)是被系统自动调用. paintEvent(QPaintEvent *)函数是QWidget类中的虚函数,用于ui的绘制,会在多种情况下被其他函 ...

  3. QT5.3无法自动调用incomingConnection函数的问题(4.7没有这个问题)

    最近将qt4.7的一个工程移到5.3,遇到了几个麻烦事,主要是这个incomingConnection监听后无法自动调用的问题,在4.7上是完全没有问题的,到了5.3就不行,网上也查了下,网友们都是放 ...

  4. 如果浏览器自动调用quirks模式打开的话

    (从已经死了一次又一次终于挂掉的百度空间人工抢救出来的,发表日期 2014-03-21) 则肯定你的html的声明,没有写好. 今天遇到几个,前面莫名其妙的多了个空格(在网页上看源码是多空格,复制到n ...

  5. PHP中 对象自动调用的方法:__set()、__get()、__tostring()

    总结: (1)__get($property_name):获取私有属性$name值时,此对象会自动调用该方法,将属性name值传给参数$property_name,通过这个方法的内部 执行,返回我们传 ...

  6. C++构造函数的自动调用(调用一个父类的构造函数,有显性调用最好,否则就默认调用无参数的构造函数)——哲学思想:不调用怎么初始化父类的成员数据和VMT?

    我总是记不住构造函数的特点,关键还是没有领会那个哲学思想:父类的构造函数一方面要初始化它自己的成员数据,另一方面也要建立它自己的VMT呀!心里默念一百遍:一定调用父类构造函数,一定调用父类构造函数,一 ...

  7. Object之魔术函数__toString() 直接输出对象引用时自动调用

    __toString()是快速获取对象的字符串信息的便捷方式 在直接输出对象引用时自动调用的方法. __toString()的作用 当我们调试程序时,需要知道是否得出正确的数据.比如打印一个对象时,看 ...

  8. QT5.3无法自动调用incomingConnection函数的问题

    最近将qt4.7的一个工程移到5.3,遇到了几个麻烦事,主要是这个incomingConnection监听后无法自动调用的问题,在4.7上是完全没有问题的,到了5.3就不行,网上也查了下,网友们都是放 ...

  9. 当使用System,out.println()打印一个对象是自动调用toString方法

    在Java中,所有的对象都是继承自Object,自然继承了toString方法,在当使用System,out.println()里面为一个对象的引用时,自动调用toString方法讲对象打印出来.如果 ...

  10. PHP 命名空间以及自动加载(自动调用的函数,来include文件)

    这篇文章的目的是记录 1. php中的自动加载函数 __autoload(), 和 spl_autoload_register()函数, 2 .php中命名空间的使用. 一.当不使用命名空间的时候 a ...

随机推荐

  1. 张高兴的 MicroPython 入门指南:(二)GPIO 的使用

    目录 什么是 GPIO 使用方法 使用微动开关点亮板载 LED 硬件需求 电路 代码 参考 什么是 GPIO GPIO 是 General Purpose Input Output 的缩写,即&quo ...

  2. 【一天一点.NET小知识】运用向量Vector<T>加速求和计算

    随着 .NET 版本的演进,从 .NET Standard 2.0 版本开始,支持 Vector<T> 类型. Vector<T> 类型:表示指定数值类型(适用于并行算法的低级 ...

  3. 洛谷P3009

    #include<iostream> #include<utility> using namespace std; typedef long long ll; #define ...

  4. Java-Response对象设置响应消息

    功能:设置响应消息 1.设置响应行 格式:HTTP/1.1 200 OK 设置状态码:setStatus(int sc) 2.设置响应头:setHeader(String name,String va ...

  5. webpack4.15.1 学习笔记(三) — 模块热替换HMR

    目录 模块热替换 HMR HMR监听文件变化 HMR 修改样式表 模块热替换 HMR 允许在运行时更新各种模块,而无需进行完全刷新.不适用于生产环境,意味着应当只在开发环境使用.启用HMR实际上就是更 ...

  6. oeasy教您玩转vim - 91 - # vim脚本编程展望

    ​ vim脚本编程展望 回忆 上次我们彻底研究了vim高亮的原理 各种语法项syntax item 关键字keyword 匹配模式match 区域region 定义好了之后还可以设置链接成组 hi d ...

  7. oeasy教您玩转vim - 81 - # 宏macro的进阶

    ​ 宏的进阶 macro 回忆 关于宏,上次有4个要点 qa 开始录制宏 q 结束录制宏 @a 应用宏 qA 追加录制宏 甚至可以编辑宏 "ap 把宏作为文本粘贴出来 编辑之后 " ...

  8. [WPF] 脱机环境实现支持拼音模糊搜索的AutoCompleteBox

    AutoCompleteBox是一个常见的提高输入效率的组件,很多WPF的第三方控件库都提供了这个组件,但基本都是字符串的子串匹配,不支持拼音模糊匹配,例如无法通过输入ldh或liudehua匹配到刘 ...

  9. 如何在AS中实现mysql查询并输出在视图上

    新建子线程启用mysql new Thread(){ @override public void run(){ //在这里进行数据库调用 } }.start(); handler简单使用方法 hand ...

  10. CentOS-7离线安装perl

    1.下载相关安装包 CentOS-7 所有rpm包的仓库地址:https://vault.centos.org/7.9.2009/os/x86_64/Packages/ perl-5.16.3-297 ...