try-with-resources语句是声明了一个或多个资源的try语句块。在java中资源作为一个对象,在程序完成后必须关闭。try-with-resources语句确保每个资源在语句结束时关闭。只要是实现了java.lang.AutoCloseable的任何对象(包括实现java.lang.Closeable的所有对象)都可以使用该方式对资源进行关闭。
在java 7之前,一般在进行文件IO操作时都需要显式的进行文件流(也可以理解为资源)的close操作,无论是操作到文件流末尾还是发生异常。往往很简单的一个逻辑都要好几行的代码进行修饰,使得代码结构变的复杂。如下例子,不管try语句块正常结束还是发生异常,都可以使用finally语句块来确保资源被关闭:
static String ReadFile(String file) throws IOException { BufferedReader br = new BufferedReader(new FileReader(file)); try { return br.readLine(); } finally { if (br != null) br.close(); } }
对于以上语句块,改写为TWR时,如下:
static String ReadFile(String file) throws IOException { try(BufferedReader br = new BufferedReader(new FileReader(file))) { return br.readLine(); } }
可以很明显看出代码的精简。
上边说过,只要是实现了AutoCloseable和Closeable接口的的对象都可以使用该方式,看下上例子中的BufferedReader类的源码实现:
public class BufferedReader extends Reader { }
其继承了Reader抽象类,继续往上查看Reader抽象类,确实实现了Closeable:
public abstract class Reader implements Readable, Closeable { }
明白了TWR具体含义,实例运用下,实现文件拷贝实现,代码中采用了两种方式实现文件拷贝过程(普通方式和NIO方式),两者存在一些性能的差异,感兴趣的读者可以自己测试代码性能。
package javaFile.copyfile; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.nio.ByteBuffer; import java.nio.channels.FileChannel; public final class CopyFile { //使用final定义工具类,不用实例化 private CopyFile() { //若实例化,则报错 throw new AssertionError(); } public static void fileCopy(String source, String target) throws IOException { try(InputStream in = new FileInputStream(source)) { try(OutputStream out = new FileOutputStream(target)){ byte[] buffer = new byte[4096]; int bytesToRead; while((bytesToRead = in.read(buffer)) != -1) { out.write(buffer, 0, bytesToRead); } } } } public static void fileCopyNIO(String source, String target) throws IOException { try(FileInputStream in = new FileInputStream(source)) { try(FileOutputStream out = new FileOutputStream(target)) { FileChannel inChannel = in.getChannel(); FileChannel outChannel = out.getChannel(); ByteBuffer buffer = ByteBuffer.allocate(4096); //申请4096字节缓冲 while(inChannel.read(buffer) != -1) { buffer.flip(); //反转此缓冲区,设置当前位置指针为0,read读文件后文件指针在缓冲区末尾,需要使用flip重置 outChannel.write(buffer); buffer.clear(); //清空缓冲区 } } } } public static void main(String[] args) throws IOException { //CopyFile copyfile = new CopyFile(); long start = System.currentTimeMillis(); CopyFile.fileCopyNIO("E:\\大数据.rar", "E:\\testtest"); long end = System.currentTimeMillis(); System.out.println("耗时:"+(end-start)); } }
原文来自:http://blog.csdn.net/ty_laurel/article/details/60348495
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY