你还在使用 try-catch-finally 关闭资源?

简介: 代码一定得写的优雅一点!你还在使用try-catch-finally关闭资源吗,如果是,那么就有点out了。皮皮甜手把手教你使用JDK7引用的try-with-resourceJDK7之前资源的关闭姿势:

代码一定得写的优雅一点!

你还在使用try-catch-finally关闭资源吗,如果是,那么就有点out了。皮皮甜手把手教你使用JDK7引用的try-with-resource

JDK7之前资源的关闭姿势:

/**  
 * jdk7以前关闭流的方式  
 *  
 * @author hetiantian  
 * */  
public class CloseResourceBefore7 {  
    private static final String FileName = "file.txt";  
    public static void main(String[] args) throws IOException {  
        FileInputStream inputStream = null;  
        try {  
            inputStream = new FileInputStream(FileName);  
            char c1 = (char) inputStream.read();  
            System.out.println("c1=" + c1);  
        } catch (IOException e) {  
            e.printStackTrace();  
        } finally {  
            if (inputStream != null) {  
                inputStream.close();  
            }  
        }  
    }  
}  

JDK7及以后关闭资源的正确姿势


try-with-resource Resource的定义:


所有实现了 java.lang.AutoCloseable[1] 接口(其中,它包括实现了 java.io.Closeable[2] 的所有对象),可以使用作为资源。简单Demo进行证实:实现java.lang.AutoCloseable接口的Resource类:

/**  
 * 资源类  
 *  
 * @author hetiantian  
 * */  
public class Resource implements AutoCloseable {  
    public void sayHello() {  
        System.out.println("hello");  
    }  
    @Override  
    public void close() throws Exception {  
        System.out.println("Resource is closed");  
    }  
} 

测试类CloseResourceIn7.java

/**  
 * jdk7及以后关闭流的方式  
 *  
 * @author hetiantian  
 * */  
public class CloseResourceIn7 {  
    public static void main(String[] args) {  
        try(Resource resource = new Resource()) {  
            resource.sayHello();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
}  

打印结果:

hello  
Resource is closed

当存在多个打开资源的时候:资源二Resource2.java

/**  
 * 资源2  
 *  
 * @author hetiantian  
 * */  
public class Resource2 implements AutoCloseable {  
    public void sayhello() {  
        System.out.println("Resource say hello");  
    }  
    @Override  
    public void close() throws Exception {  
        System.out.println("Resource2 is closed");  
    }  
}  

测试类CloseResourceIn7.java

/**  
 * jdk7及以后关闭流的方式  
 *  
 * @author hetiantian  
 * */  
public class CloseResourceIn7 {  
    public static void main(String[] args) {  
        try(Resource resource = new Resource(); Resource2 resource2 = new Resource2()) {  
            resource.sayHello();  
            resource2.sayhello();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
}  

打印结果:

`hello  
Resource say hello  
Resource2 is closed  
Resource is closed  
`

即使资源很多,代码也可以写的很简洁,如果用JDK7之前的方式去关闭资源,那么资源越多,用fianl关闭资源时嵌套也就越多。最近写的这篇:写了个全局变量的bug,也推荐看下。


那么它的底层原理又是怎样的呢,由皮皮甜独家揭秘优雅关闭资源背后的密码秘密


查看编译的class文件CloseResourceIn7.class:

public class CloseResourceIn7 {  
    public CloseResourceIn7() {  
    }  
    public static void main(String[] args) {  
        try {  
            Resource resource = new Resource();  
            Throwable var2 = null;  
            try {  
                resource.sayHello();  
            } catch (Throwable var12) {  
                var2 = var12;  
                throw var12;  
            } finally {  
                if (resource != null) {  
                    if (var2 != null) {  
                        try {  
                            resource.close();  
                        } catch (Throwable var11) {  
                            var2.addSuppressed(var11);  
                        }  
                    } else {  
                        resource.close();  
                    }  
                }  
            }  
        } catch (Exception var14) {  
            var14.printStackTrace();  
        }  
    }  
}  

可以发现编译以后生成了try-catch-finally语句块 finally中的var2.addSuppressed(var11);

是不是有疑问?其实这么做是为了处理异常屏蔽的,我们将代码修改一下。

资源Resource.java

/**  
 * 资源类  
 *  
 * @author hetiantian  
 * */  
public class Resource implements AutoCloseable {  
    public void sayHello() throws Exception {  
        throw new Exception("Resource throw Exception");  
    }  
    @Override  
    public void close() throws Exception {  
        throw new Exception("Close method throw Exception");  
    }  
}  

两个方法里面都抛出异常

测试类CloseResourceIn7.java

/**  
 * jdk7及以后关闭流的方式  
 *  
 * @author hetiantian  
 * */  
public class CloseResourceIn7 {  
    public static void main(String[] args) {  
        try {  
            errorTest();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
    private static void errorTest() throws Exception {  
        Resource resource = null;  
        try {  
            resource = new Resource();  
            resource.sayHello();  
        }  
        finally {  
            if (resource != null) {  
                resource.close();  
            }  
        }  
    }  
}  

打印结果:

java.lang.Exception: Close method throw Exception  
    at com.shuwen.Resource.close(Resource.java:15)  
    at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:27)  
    at com.shuwen.CloseResourceIn7.main(CloseResourceIn7.java:12)  

只打印了最后出现的异常【异常屏蔽】这样会给开发人员排查错误带来一定的困难 我们换成try-with-resource方法实现CloseResourceIn7.java

/**  
 * jdk7及以后关闭流的方式  
 *  
 * @author hetiantian  
 * */  
public class CloseResourceIn7 {  
    public static void main(String[] args) {  
        try {  
            errorTest();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
    private static void errorTest() throws Exception {  
        try(Resource resource = new Resource()) {  
            resource.sayHello();  
        }  
    }  
}  

打印信息:

java.lang.Exception: Resource throw Exception  
    at com.shuwen.Resource.sayHello(Resource.java:10)  
    at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:20)  
    at com.shuwen.CloseResourceIn7.main(CloseResourceIn7.java:12)  
    Suppressed: java.lang.Exception: Close method throw Exception  
        at com.shuwen.Resource.close(Resource.java:15)  
        at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:21)  
        ... 1 more

可以发现,异常信息中多了一个Suppressed的提示,告诉我们这个异常其实由两个异常组成,Close method throw Exception这个异常是被Suppressed【屏蔽】的异常

相关文章
|
云安全 域名解析 安全
警惕主动外联!云防火墙检测拦截勒索、Muhstik僵尸网络等 Log4j2漏洞利用
近期,阿里云安全观测到,在 Apache Log4j2 漏洞攻击全程中,无论是在漏洞利用阶段,还是后续要进行验证和进一步的控制利用,大多涉及多次受害服务器的主动外联,云防火墙已陆续发现并拦截60余万次涉及勒索、挖矿家族的漏洞利用行为。
1283 0
警惕主动外联!云防火墙检测拦截勒索、Muhstik僵尸网络等 Log4j2漏洞利用
el-tree技巧之只能选中最后一层级的子节点以及查找树结构第一个无子节点的叶节点
el-tree技巧之只能选中最后一层级的子节点以及查找树结构第一个无子节点的叶节点
|
9月前
|
算法
重磅!2025年中科院预警期刊名单正式发布!
中国科学院文献情报中心定期发布《国际期刊预警名单》,旨在防范学术不端和不当出版行为。2025年最新名单聚焦两大问题:一是引用操纵、论文工厂等破坏科研生态的行为;二是中国作者占比过高或APC费用不合理,影响学术成果国际化传播。自2022年起,预警名单调整至年初发布,便于科研人员规划投稿。名单结合定量数据与专家评估,动态反映期刊风险。被列预警期刊可能影响职称评审及科研经费认可,建议优先选择中科院分区表推荐期刊,警惕快速代发陷阱,并关注期刊官网声明。未来,强化学术自律和技术工具应用将助力科研规范化,推动中国学术走向全球。
882 0
|
搜索推荐 关系型数据库 MySQL
mysql like查询优化
通过合理的索引设计、使用全文索引、优化查询结构以及考虑分片和分区表,可以显著提高MySQL中 `LIKE`查询的性能。针对不同的应用场景选择合适的优化策略,能够有效地提升数据库查询效率,减少查询时间。希望这些方法和技巧能帮助您优化MySQL数据库中的模糊查询。
1321 4
|
存储 Linux
小白带你走进linux文件权限与归档压缩(十)
小白带你走进linux文件权限与归档压缩(十)
164 0
|
存储 缓存 小程序
小程序数据缓存机制应用
小程序数据缓存机制应用
396 0
小程序数据缓存机制应用
|
小程序
狼人杀微信小程序项目实例(附源码)
狼人杀是一款多人参与的,通过语言描述推动、较量口才和分析判断能力的策略类桌面游戏
884 0
狼人杀微信小程序项目实例(附源码)
|
机器学习/深度学习 人工智能 语音技术
AI实现语音文字处理,PaddleSpeech项目安装使用 | 机器学习
AI实现语音文字处理,PaddleSpeech项目安装使用 | 机器学习
AI实现语音文字处理,PaddleSpeech项目安装使用 | 机器学习
|
网络协议 Java Android开发
基于Kotlin实现一个简单的TCP自定义协议
一. 开发背景 想要成为一名优秀的Android开发,你需要一份完备的 知识体系,在这里,让我们一起成长为自己所想的那样~。
基于Kotlin实现一个简单的TCP自定义协议
|
算法
数学建模 线性规划(LP)
数学建模 线性规划(LP)
298 0