java 压缩与解压

时间:2023-03-09 04:26:26
java 压缩与解压

最近复习到IO,想找个案例做一做,恰好下载了许多图片压缩包,查看图片很不方便,所以打算用IO把图片都解压到同一个文件夹下。然后集中打包。

本例使用jdk自带的ZipInputStream和ZipOutPutStream,功能有限不支持rar但是api很简单。

import java.io.*;
import java.util.zip.*; /**
* Created by tm on 2017/2/18.
* time : 17:33
* project_name : toolSite
* 提供zip文件和文件夹的解压和压缩功能。
*/
public class ZipTool { /**
* 压缩单个文件
* 此处传参都是绝对路径。
* @param src 源文件绝对路径名
* @param destPath 将生成的压缩包绝对路径(路径以/结尾)
* @param destName 将生成的压缩包文件名
* @throws IOException
*/
public static void compressFile(String src, String destPath,String destName) throws IOException {
File f = new File(src);
if (!f.exists()) {
throw new RuntimeException("file not find ");
} File folder = new File(destPath);
if(!folder.exists()){
folder.mkdir();
} InputStream is = new FileInputStream(f);
BufferedInputStream buffIn = new BufferedInputStream(is);
//创建文件File,创建文件输出流,包装到Zip文件输出处理流上,最后把Zip处理流再包装到Buffer缓冲输出流中。
ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(new File(destPath+destName)));
BufferedOutputStream buffOut = new BufferedOutputStream(zos);
//需要将文件项put到entry中,解压文件时从此entry中获取数据。
zos.putNextEntry(new ZipEntry(f.getName())); //b是指the next byte of data
int b;
while ((b = buffIn.read()) != -1) {
buffOut.write(b);
} buffIn.close();
buffOut.flush();
buffOut.close();
} /**
* 解压单个zip文件(上面压缩后的解压,认为不包含子文件夹)
* @param src 源zip的绝对路径
* @param dest 解压的目标文件夹(路径以/结尾)
* @throws IOException
*/
public static void unCompress(String src, String dest) throws IOException {
File file = new File(src);
if(!file.exists()){
throw new RuntimeException("zip file not found");
} File d = new File(dest);
if(!d.exists()){
d.mkdir();
} InputStream is = new FileInputStream(file);
ZipInputStream zis = new ZipInputStream(is);
ZipEntry entry;
while ((entry = zis.getNextEntry()) != null) {
//解压文件需要从zip中读取每一个entry,一个一个地outPut出来
OutputStream out = new FileOutputStream(new File(dest + entry.getName()));
BufferedOutputStream bos = new BufferedOutputStream(out);
BufferedInputStream buffIn = new BufferedInputStream(new ZipFile(file).getInputStream(entry));
int b;
while ((b = buffIn.read()) != -1) {
bos.write(b);
}
bos.flush();
buffIn.close();
bos.close();
}
zis.close();
} /**
* 压缩指定的文件夹(代码里没有对路径做校验,读者可自行添加,参考上例)
* @param srcFolder 源文件夹
* @param destFileName 目标文件名
* @throws IOException
*/
public static void compressNestFolder(String srcFolder, String destFileName) throws IOException {
System.out.println(">>>>>>>>>>开始压缩");
File folder = new File(srcFolder);
if (!folder.exists()) {
throw new RuntimeException("folder not find Exception");
}
OutputStream os = new FileOutputStream(new File(destFileName));
ZipOutputStream zos = new ZipOutputStream(os);
BufferedOutputStream bo = new BufferedOutputStream(zos);
zipFileOrFolder(zos, bo, "", folder);
zos.close();
bo.close();
System.out.println("压缩完成>>>>>>>>>>");
} /**
* 将目录下所有的文件和子文件都压缩到zos中
* @param zos
* @param bo
* @param src
* @param f
* @throws IOException
*/
private static void zipFileOrFolder(ZipOutputStream zos, BufferedOutputStream bo, String src, File f) throws IOException {
File[] files;
if (f.isDirectory()) {
files = f.listFiles();
assert files != null;
System.out.println("========== " + f.getName());
//添加文件夹的entry,不添加的话,解压时会出问题。
zos.putNextEntry(new ZipEntry(
src+f.getName()+"/"
));
System.out.println(
src+f.getName()+"/"
);
src = src+f.getName() + "/";
for (File f1 : files) {
//使用了递归调用获取下一层文件夹并压缩
zipFileOrFolder(zos, bo, src, f1);
}
} else {
System.out.println("---------- " + f.getName());
InputStream in = new FileInputStream(f);
BufferedInputStream bi = new BufferedInputStream(in);
zos.putNextEntry(new ZipEntry(
src+f.getName()
));
int b;
while ((b = bi.read()) != -1) {
bo.write(b);
}
bo.flush();
bi.close();
zos.closeEntry();
}
} /**
* 解压单个zip文件,保留文件夹层级关系
* @param zipFile
* @param outPath
* @throws IOException
*/
public static void unZip(String zipFile, String outPath) throws IOException {
File zip = new File(zipFile);
ZipFile zf = new ZipFile(zip);
InputStream is = new FileInputStream(zipFile);
ZipInputStream zis = new ZipInputStream(is);
ZipEntry e;
File out = new File(outPath);
if (!out.exists()) {
out.mkdir();
}
while ((e = zis.getNextEntry()) != null) {
System.out.println(e.isDirectory());
File f = new File(outPath + e.getName());
System.out.println(f.getAbsolutePath());
if(e.isDirectory()){
f.mkdir();continue;
}
OutputStream os = new FileOutputStream(f);
BufferedOutputStream bos = new BufferedOutputStream(os);
BufferedInputStream bis = new BufferedInputStream(zf.getInputStream(e));
int b;
while ((b = bis.read()) != -1) {
bos.write(b);
}
bos.flush();
bos.close();
bis.close();
}
zis.close();
} public static void unZipAll(String zipFolder,String destFolder){
//todo
} public static void main(String[] args) throws IOException {
compressFile("D:/test/123.txt", "D:/test/a/b/", "123.zip");
unCompress("D:/test/123.zip", "D:/test/");
compressNestFolder("D:/test/测试zip/新垣结衣/", "D:/test/测试zip/新垣结衣1.zip");
unZip("D:/test/测试zip/新垣结衣1.zip", "D:/test/测试zip/");
}
}

后记:

1.添加到zip中使用putNextEntry,从zip中读取时需要创建一个ZipFile然后从entry中获取InputStream。

2.需要特别注意压缩文件时的Entry存放次序,在添加entry时务必先添加文件夹的entry。因为取entry时,如果乱序就会找不到文件。比如获取/bt/123.pic时,因为没有实现创建父目录/bt,导致写文件时找不到文件。

3.对entry.isDirectory()的判断,api中是根据name是否以 / 结尾的。因此存放folder时,也要使用 / 作为路径分割符。java在windows平台上是支持 / 的,虽然windows平台上路径分隔符是 \  (java中是\\)。建议还是用 / 吧。

4.Zip对象存放文件是使用相对路径的,所以压缩什么的不要有盘符。添加目录时以 / 结尾,多级目录时添加文件使用完全相对路径名 /相对路径1/相对路径2/文件名,可以通过查看打印信息了解方法执行过程。

-----------------以上-----------------