功能需求描述:
对ZIP包中的excel文件添加水印
功能代码实现:
public static void main(String[] args) {
ZipInputStream zis = null;
ZipOutputStream zos = null;
try {
//读取文件
FileInputStream fis = new FileInputStream("C:/Users/chens/Downloads/A.zip");
zis = new ZipInputStream(fis, Charset.forName("GBK"));
//输出文件
zos = new ZipOutputStream(new FileOutputStream("C:/Users/chens/Downloads/B.zip"));
ZipEntry zipEntry;
//水印标识
String waterMark = "name-20240321";
while ((zipEntry = zis.getNextEntry()) != null) {
zos.putNextEntry(new ZipEntry(zipEntry.getName()));
//添加水印工具类
CustomCommonWaterMarkUtil.excelWaterMark(zipInputStream,zos,waterMark);
zos.flush();
zos.closeEntry();
}
} catch (Exception e) {
System.out.println(e);
} finally {
IOUtils.closeQuietly(zis);
IOUtils.closeQuietly(zos);
}
}
出现的问题:
加完水印之后的ZIP包里面的文件大小显示0KB,点击会报错,错误信息:解压文件失败!如下
但是文件打开后内容是正常的,能够正常解压,解压出来后大小显示正常
原因分析:
上面加水印的操作都没有问题,问题出在关闭文件流的顺序问题,上面是先关闭的输入流再关闭的输出流,此时可能就会存在潜在的数据丢失或资源泄漏问题。改成先关闭输出流,然后再关闭输入流后问题解决。
总结:
在Java中,关闭文件流的顺序很重要,特别是当你使用多个流(如缓冲流和底层流)时。如果不正确地关闭这些流,可能会导致资源泄漏、数据丢失或其他不可预见的问题。
以下是一些关于关闭Java文件流的最佳实践和注意事项:
- 底层流优先关闭:如果你使用了缓冲流(如
BufferedOutputStream
或BufferedReader
)来包装另一个底层流(如FileOutputStream
或FileReader
),你应该首先关闭缓冲流。当你关闭缓冲流时,它会自动关闭其底层的流。如果你先关闭了底层流,而缓冲流仍然试图写入数据,那么就会出现问题。
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("file.txt"));
// ... 使用bos进行写操作 ...
bos.close(); // 这会同时关闭底层的FileOutputStream
- 顺序问题:关闭文件流的正确顺序是首先关闭输出流,然后再关闭输入流。这样可以确保数据被正确保存和释放,并且能够避免潜在的数据丢失或资源泄漏问题。
- 使用try-with-resources语句:从Java 7开始,你可以使用try-with-resources语句来自动管理资源的关闭。这可以确保在try块执行完毕后,所有实现了
AutoCloseable
接口的资源(包括大多数流)都会被自动关闭。
try (BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("file.txt"))) {
// ... 使用bos进行写操作 ...
} // try块结束时,bos会被自动关闭
- 避免多次关闭:一旦一个流被关闭,再次尝试关闭它通常会导致异常(通常是
IOException
)。使用try-with-resources语句可以很好地避免这个问题,因为它确保了每个资源只会被关闭一次。 - 检查异常:在关闭流时,可能会抛出
IOException
。你应该检查这些异常并适当地处理它们,以确保你的程序能够优雅地处理资源关闭失败的情况。 - 关闭所有流:即使你只使用了一个流,也应该确保它被正确关闭。未关闭的流可能会导致文件句柄泄漏,这在长时间运行的程序中可能会成为一个问题。
- ByteArrayOutputStream或ByteArrayInputStream是内存读写流,这个和其他流不一样,看源码就能看到这个是使用字节数组读内存的,这个字节数组是它的成员变量,当这个数组不再使用变成垃圾的时候,Java的垃圾回收机制会将它回收。所以不需要关流。
总之,正确地关闭Java文件流是非常重要的,有助于保持代码的可靠性和可维护性。遵循上述最佳实践可以帮助你避免许多常见的问题。