首先,我们如果要对Zip文件进行解析,就要获取Zip文件。通常获取Zip文件就要引入ZipInputStream。
在这里要注意,ZipInputStream是FilterInputStream的一个间接子类,所以ZipInputStream是具有额外附加功能的InputStream,其并不仅具有真正的读取操作。
读取Zip包
所以,我们首先一个ZipInputStream,在他里面再引入一个FilterInputStream做为数据源,然后再循环调用getNextEntry(),直至返回值为null,就代表zip流结束。
// 2.解析读取zip文件
try (ZipInputStream in = new ZipInputStream(new FileInputStream(zipFile),Charset.forName("gbk"))) {
// 遍历zip文件中的每个子文件
ZipEntry zipEntry = null;
while((zipEntry = in.getNextEntry()) != null) {
// 获取zip压缩包中的子文件名称
String zipEntryFileName = zipEntry.getName();
String zipFilePath = targetDir.getPath() + "\\" + zipEntryFileName;
// 创建该文件的输出流
try(BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(zipFilePath))){
// 读取该子文件的字节内容
byte[] buff = new byte[1024];
int len = -1;
while((len = in.read(buff)) != -1) {
bos.write(buff, 0, len);
}
}
}
} catch (IOException e) {
e.printStackTrace();
}
在此代码段里,一个zipEntry代表一个压缩文件或者目录,直到读取的返回值为null时,代表读取结束。读取其字节内容时,使用read()方法,直到返回值为-1,表示读取完毕。
生成Zip文件
要生成一个Zip文件,首先要创建一个ZipOutputStream,再包装一个FilterOutputStream,(ZipOutputStream是FilterOutputStream的子类,它可以直接将内容写入到zip包)然后再每写入一个文件前,调用putNextEntry(),再用write()写入byte[]数据,写入完成之后最后再调用closeEntry()结束这个文件的打包。
// 原始目录
File rootDir = new File("d:\\io\\1");
// 将原始目录中的文件,"写入"zip压缩文件
try (ZipOutputStream out = new ZipOutputStream(new FileOutputStream(rootDir.getParent() + "\\" + rootDir.getName() + ".zip"))) {
// 获取并遍历原始目录下的子文件列表
File[] files = rootDir.listFiles();
for(File f : files) {
// 创建一个ZipEntry
out.putNextEntry(new ZipEntry(f.getName()));
// 将原文件的字节内容,写入Zip压缩包
out.write(Files.readAllBytes(f.toPath()));
// 结束当前ZipEntry
out.closeEntry();
}
} catch (IOException e) {
e.printStackTrace();
}
总结
ZipInputStream可以读取zip格式的流,ZipOutputStream可以将多份数据写入zip包。
在读写zip文件的时候要配合FilterInputStream和FilterOutputStream同时进行。