java - 处理目录和文件时处理 IO 异常的最佳实践?
问题描述
示例代码如下。它将目标文件和目录从一个位置复制到另一个位置。在跨网络处理文件时,处理 IO 异常的最佳实践是什么?
我使用了 printStackTrace() 但觉得这只是一个更好的解决方案的占位符。正在记录答案吗?除了记录之外,是否应该有另一个步骤来实际“处理”错误?
感谢您的反馈。
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
/**
This is a test program to copy a directory(s) & file(s) from one location to another.
*/
public class CopyTest{
public static void main(String[] args) {
//Declarations
String sourcePath = "I:\\MB\\PO";
String destPath = "C:\\testPO\\";
System.out.println("Source path: " + sourcePath);
System.out.println("Destination path: " + destPath);
File source = new File(sourcePath);
File dest = new File(destPath);
//Process
//Call to method copyUsingStream
long start = System.nanoTime(); //start recording how much time the copy takes.
copyUsingStream(source, dest); //method to copy the directory/files.
System.out.println("Time taken to copy the file: "+(System.nanoTime() -start) + " nanoseconds");
} //end main method
/**
The copyUsingStream method is a recursive method to copy folders and files from one location to another.
*/
private static void copyUsingStream(File source, File dest) {
if (!source.isDirectory()){
// If source is a file -> copy it to the new folder
InputStream inStream = null;
OutputStream outStream = null;
try {
inStream = new FileInputStream(source);
outStream = new FileOutputStream(dest);
byte[] buffer = new byte[1024];
int length;
while ((length = inStream.read(buffer)) > 0) {
outStream.write(buffer, 0, length);
}
} catch(IOException ioe) {
ioe.printStackTrace();
} finally {
try{
inStream.close();
outStream.close();
System.out.println("File copied from " + source + " to " + dest + "successfully");
} catch(IOException ioe2) {
ioe2.printStackTrace();
}
}
} else {
//If a directory -> create the directory inside the new destination
//List all contents
if (!dest.exists()) {
dest.mkdir();
System.out.println("Directory copied from " + source + " to " + dest + "successfully");
}
String folder_contents[] = source.list();
for (String file : folder_contents) {
File srcFile = new File(source, file);
File destFile = new File(dest, file);
copyUsingStream(srcFile, destFile);
}
}
} //end method copyUsingStream
} //end class CopyTest
没有捕获的方法:
private static void copyUsingStream(File source, File dest) throws IOException {
if (!source.isDirectory()){
// If source is a file -> copy it to the new folder
InputStream inStream = null;
OutputStream outStream = null;
try {
inStream = new FileInputStream(source);
outStream = new FileOutputStream(dest);
byte[] buffer = new byte[1024];
int length;
while ((length = inStream.read(buffer)) > 0) {
outStream.write(buffer, 0, length);
}
} finally {
inStream.close();
outStream.close();
System.out.println("File copied from " + source + " to " + dest + "successfully");
}
} else {
//If a directory -> create the directory inside the new destination
//List all contents
if (!dest.exists()) {
dest.mkdir();
System.out.println("Directory copied from " + source + " to " + dest + "successfully");
}
String folder_contents[] = source.list();
for (String file : folder_contents) {
File srcFile = new File(source, file);
File destFile = new File(dest, file);
copyUsingStream(srcFile, destFile);
}
}
} //end method copyUsingStream
解决方案
这在很大程度上取决于您的应用程序。
无论如何都继续运行的应用程序(例如 Web 服务器、守护程序和批处理器)通常会在文件中记录此类错误以及时间戳、线程 ID 和可能的其他有用信息。
我对两个日志文件的组合有很好的经验。
myapp.log
只接收重要消息,通常是警告和错误。此文件适用于普通用户和系统操作员。debug.log
是给开发者的。它提供错误发生前的调试消息,但只要一切正常,就没有消息。要启用此功能,需要一个内存缓冲区。
如果您对该缓冲区感兴趣,可以查看http://stefanfrings.de/bfUtilities/index.html。该网站是德文的,但图书馆和它的文档是英文的。
在桌面 GUI 应用程序中,当错误中止请求的操作时,最好在弹出窗口中显示简短的错误消息,并在可扩展框中隐藏详细信息(堆栈跟踪)。不要忘记清楚地告诉用户什么操作失败了。异常本身对您的开发人员来说可能已经很清楚了,但普通用户期望的技术文本较少。例如:“从服务 weather.com 加载天气信息失败:连接失败”,后跟堆栈跟踪。
对于立即停止的控制台应用程序,我更喜欢直接在屏幕上查看堆栈跟踪,如printStackTrace()
.
推荐阅读
- graphics - Godot 中的纹理没有 100% 显示
- reactjs - 从后台计时器反应重定向浏览器
- android - 升级到 Android Studio Artic Fox 和 AGP 7.0.0 后,自定义视图不再在 Layout Preview 中呈现
- c# - Xamarin Forms IOS Google Sheets API:找不到client_secret.json文件
- javascript - 防止 redux 将重复的商品添加到购物车
- javascript - Swiper 和 Javascript:使用外部按钮转到特定幻灯片
- java - 如何在 Java 中使用 filter() 从 arrayList 中过滤字符串日期范围?
- video - 如何使用 Azure 媒体服务对 mp4、OGG 格式的视频文件应用水印
- android - 拆分成模块后,Jenkins 中的 Android 构建速度太慢
- image - Flutter:如何选择和显示图像