代码评审——程序中流、连接等关闭问题

2024-03-07 09:40

本文主要是介绍代码评审——程序中流、连接等关闭问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问题描述:

在程序开发过程中,必然面临着与文件、数据库等进行交互。使用流等方式来建立连接,并进行操作,也成为了开发者的日常。但在开发过程中,对连接的回收、关闭等操作的管理,经常会被遗忘,从而引发问题。

可以看下这个举例的代码:

Connection con;
try {con = JDBC.createConnection("XXXX", prop);String query = "SELECT * FROM TABLENAME";Statement stat = con.createStatement();ResultSet res = stat.executeQuery(query);res.close();stat.close();
} catch (SQLException e) {e.printStackTrace();
} finally{if(stat != null){stat.close();}if(con != null){con.close();}
}

此时,使用静态代码扫描工具,会出现以下提示:
Use try-with-resources or close this “ResultSet” in a “finally” clause.

在这里插入图片描述


原因分析:

静态工具代码扫描工具给出的解释为:

Connections, streams, files, and other classes that implement the Closeable interface or its super-interface, AutoCloseable, needs to be closed after use. Further, that close call must be made in a finally block otherwise an exception could keep the call from being made. Preferably, when class implements AutoCloseable, resource should be created using “try-with-resources” pattern and will be closed automatically.
Failure to properly close resources will result in a resource leak which could bring first the application and then perhaps the box it’s on to their knees.

其实,从开发者的角度来说,这个问题非常常见,处理也很简单。但是经常会忽视这个问题,可能是遗忘,可能是漏写。上面的示例代码就可以看出,是漏写了,仅关闭了Statement和Connection。如果不进行关闭,非常容易造成资源的泄露。


解决方案:

解决方案也很简单,只需要在finally中增加对ResultSet的关闭即可。

Connection con;
try {con = JDBC.createConnection("XXXX", prop);String query = "SELECT * FROM TABLENAME";Statement stat = con.createStatement();ResultSet res = stat.executeQuery(query);res.close();stat.close();
} catch (SQLException e) {e.printStackTrace();
} finally{if(res != null){res.close();}if(stat != null){stat.close();}if(con != null){con.close();}
}

项目实际意义:

在项目开发过程中,最常使用的连接包括以下3项:

  • 文件流,包括字节流、字符流等等。
  • 数据库连接,redis等连接。
  • HTTP连接。

不关闭连接,很容易造成严重的结果。

  • 针对文件流等,如果不关闭,会造成文件写入错误,别的文件也无法读取这个文件。
  • 针对数据库连接,如果不关闭,会直接造成数据库的连接数暴涨(和访问这个服务的次数有关),最终造成数据库无法读写。虽然大多数情况下,会由对应的连接池(DB连接池或Redis连接池),来进行回收,但在业务实现上,也存在需要自己构建连接的情况。此时不关闭,甚至会造成项目的大型事故
  • 针对HTTP连接的情况,需要根据实际情况。如果需要不断交互,就不需要直接回收,否则就需要回收。

后面两种情况,因为有连接池的存在,经常会由连接池来进行销毁或回收等。

开发者在处理上述连接的过程中,如果不使用连接池等组件的情况,一定要及时关闭,否则很容易引起事故。

这篇关于代码评审——程序中流、连接等关闭问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/783092

相关文章

VSCode中C/C++编码乱码问题的两种解决方法

《VSCode中C/C++编码乱码问题的两种解决方法》在中国地区,Windows系统中的cmd和PowerShell默认编码是GBK,但VSCode默认使用UTF-8编码,这种编码不一致会导致在VSC... 目录问题方法一:通过 Code Runner 插件调整编码配置步骤方法二:在 PowerShell

mybatis-plus分页无效问题解决

《mybatis-plus分页无效问题解决》本文主要介绍了mybatis-plus分页无效问题解决,原因是配置分页插件的版本问题,旧版本和新版本的MyBatis-Plus需要不同的分页配置,感兴趣的可... 昨天在做一www.chinasem.cn个新项目使用myBATis-plus分页一直失败,后来经过多方

mybatis-plus 实现查询表名动态修改的示例代码

《mybatis-plus实现查询表名动态修改的示例代码》通过MyBatis-Plus实现表名的动态替换,根据配置或入参选择不同的表,本文主要介绍了mybatis-plus实现查询表名动态修改的示... 目录实现数据库初始化依赖包配置读取类设置 myBATis-plus 插件测试通过 mybatis-plu

使用Dify访问mysql数据库详细代码示例

《使用Dify访问mysql数据库详细代码示例》:本文主要介绍使用Dify访问mysql数据库的相关资料,并详细讲解了如何在本地搭建数据库访问服务,使用ngrok暴露到公网,并创建知识库、数据库访... 1、在本地搭建数据库访问的服务,并使用ngrok暴露到公网。#sql_tools.pyfrom

Java springBoot初步使用websocket的代码示例

《JavaspringBoot初步使用websocket的代码示例》:本文主要介绍JavaspringBoot初步使用websocket的相关资料,WebSocket是一种实现实时双向通信的协... 目录一、什么是websocket二、依赖坐标地址1.springBoot父级依赖2.springBoot依赖

如何用java对接微信小程序下单后的发货接口

《如何用java对接微信小程序下单后的发货接口》:本文主要介绍在微信小程序后台实现发货通知的步骤,包括获取Access_token、使用RestTemplate调用发货接口、处理AccessTok... 目录配置参数 调用代码获取Access_token调用发货的接口类注意点总结配置参数 首先需要获取Ac

讯飞webapi语音识别接口调用示例代码(python)

《讯飞webapi语音识别接口调用示例代码(python)》:本文主要介绍如何使用Python3调用讯飞WebAPI语音识别接口,重点解决了在处理语音识别结果时判断是否为最后一帧的问题,通过运行代... 目录前言一、环境二、引入库三、代码实例四、运行结果五、总结前言基于python3 讯飞webAPI语音

什么是 Java 的 CyclicBarrier(代码示例)

《什么是Java的CyclicBarrier(代码示例)》CyclicBarrier是多线程协同的利器,适合需要多次同步的场景,本文通过代码示例讲解什么是Java的CyclicBarrier,感... 你的回答(口语化,面试场景)面试官:什么是 Java 的 CyclicBarrier?你:好的,我来举个例

IDEA连接达梦数据库的详细配置指南

《IDEA连接达梦数据库的详细配置指南》达梦数据库(DMDatabase)作为国产关系型数据库的代表,广泛应用于企业级系统开发,本文将详细介绍如何在IntelliJIDEA中配置并连接达梦数据库,助力... 目录准备工作1. 下载达梦JDBC驱动配置步骤1. 将驱动添加到IDEA2. 创建数据库连接连接参数

Flask解决指定端口无法生效问题

《Flask解决指定端口无法生效问题》文章讲述了在使用PyCharm开发Flask应用时,启动地址与手动指定的IP端口不一致的问题,通过修改PyCharm的运行配置,将Flask项目的运行模式从Fla... 目录android问题重现解决方案问题重现手动指定的IP端口是app.run(host='0.0.