Java压缩技术(三) ZIP解压缩——Java原生实现

JavaEye的朋友跟我说:“你一口气把ZIP压缩和解压缩都写到一个帖子里,我看起来很累,不如分开好阅读”。ok,面向读者需求,我做调整,这里单说ZIP解压缩!

相关链接:
[url=http://snowolf.iteye.com/blog/465433]Java压缩技术(一) ZLib[/url]
[url=http://snowolf.iteye.com/blog/642298]Java压缩技术(二) ZIP压缩——Java原生实现[/url]
[url=http://snowolf.iteye.com/blog/642492]Java压缩技术(三) ZIP解压缩——Java原生实现[/url]
[url=http://snowolf.iteye.com/blog/643010]Java压缩技术(四) GZIP——Java原生实现[/url]
[url=http://snowolf.iteye.com/blog/643443]Java压缩技术(五) GZIP相关——浏览器解析[/url]
[url=http://snowolf.iteye.com/blog/644591]Java压缩技术(六) BZIP2——Commons实现[/url]
[url=http://snowolf.iteye.com/blog/648652]Java压缩技术(七) TAR——Commons实现[/url]


解压缩与压缩运作方式相反,原理大抵相同,由ZipInputStream通过read方法对数据解压,同时需要通过CheckedInputStream设置冗余校验码,如:


CheckedInputStream cis = new CheckedInputStream(new FileInputStream(
srcFile), new CRC32());

ZipInputStream zis = new ZipInputStream(cis);


需要注意的是,在构建解压文件时,需要考虑目录的自动创建,这里通过递归方式逐层创建父目录,如下所示:


/**
* 文件探针
*
*
* 当父目录不存在时,创建目录!
*
*
* @param dirFile
*/
private static void fileProber(File dirFile) {

File parentFile = dirFile.getParentFile();
if (!parentFile.exists()) {

// 递归寻找上级目录
fileProber(parentFile);

parentFile.mkdir();
}

}


在压缩的时候,我们是将一个一个文件作为压缩添加项(ZipEntry)添加至压缩包中,解压缩就要将一个一个压缩项从压缩包中提取出来,如下所示:


/**
* 文件 解压缩
*
* @param destFile
* 目标文件
* @param zis
* ZipInputStream
* @throws Exception
*/
private static void decompress(File destFile, ZipInputStream zis)
throws Exception {

ZipEntry entry = null;
while ((entry = zis.getNextEntry()) != null) {

// 文件
String dir = destFile.getPath() + File.separator + entry.getName();

File dirFile = new File(dir);

// 文件检查
fileProber(dirFile);

if (entry.isDirectory()){
dirFile.mkdirs();
} else {
decompressFile(dirFile, zis);
}

zis.closeEntry();
}
}



最核心的解压缩实现,其实与压缩实现非常相似,代码如下所示:


/**
* 文件解压缩
*
* @param destFile
* 目标文件
* @param zis
* ZipInputStream
* @throws Exception
*/
private static void decompressFile(File destFile, ZipInputStream zis)
throws Exception {

BufferedOutputStream bos = new BufferedOutputStream(
new FileOutputStream(destFile));

int count;
byte data[] = new byte[BUFFER];
while ((count = zis.read(data, 0, BUFFER)) != -1) {
bos.write(data, 0, count);
}

bos.close();
}



来个完整的解压缩实现,代码如下:


/**
* 2010-4-12
*/
package org.zlex.commons.io;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.zip.CRC32;
import java.util.zip.CheckedInputStream;
import java.util.zip.CheckedOutputStream;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
import java.util.zip.ZipOutputStream;

/**
* ZIP压缩工具
*
* @author 梁栋
* @since 1.0
*/
public class ZipUtils {

public static final String EXT = ".zip";
private static final String BASE_DIR = "";
private static final String PATH = File.separator;
private static final int BUFFER = 1024;

/**
* 文件 解压缩
*
* @param srcPath
* 源文件路径
*
* @throws Exception
*/
public static void decompress(String srcPath) throws Exception {
File srcFile = new File(srcPath);

decompress(srcFile);
}

/**
* 解压缩
*
* @param srcFile
* @throws Exception
*/
public static void decompress(File srcFile) throws Exception {
String basePath = srcFile.getParent();
decompress(srcFile, basePath);
}

/**
* 解压缩
*
* @param srcFile
* @param destFile
* @throws Exception
*/
public static void decompress(File srcFile, File destFile) throws Exception {

CheckedInputStream cis = new CheckedInputStream(new FileInputStream(
srcFile), new CRC32());

ZipInputStream zis = new ZipInputStream(cis);

decompress(destFile, zis);

zis.close();

}

/**
* 解压缩
*
* @param srcFile
* @param destPath
* @throws Exception
*/
public static void decompress(File srcFile, String destPath)
throws Exception {
decompress(srcFile, new File(destPath));

}

/**
* 文件 解压缩
*
* @param srcPath
* 源文件路径
* @param destPath
* 目标文件路径
* @throws Exception
*/
public static void decompress(String srcPath, String destPath)
throws Exception {

File srcFile = new File(srcPath);
decompress(srcFile, destPath);
}

/**
* 文件 解压缩
*
* @param destFile
* 目标文件
* @param zis
* ZipInputStream
* @throws Exception
*/
private static void decompress(File destFile, ZipInputStream zis)
throws Exception {

ZipEntry entry = null;
while ((entry = zis.getNextEntry()) != null) {

// 文件
String dir = destFile.getPath() + File.separator + entry.getName();

File dirFile = new File(dir);

// 文件检查
fileProber(dirFile);

if (entry.isDirectory()) {
dirFile.mkdirs();
} else {
decompressFile(dirFile, zis);
}

zis.closeEntry();
}
}

/**
* 文件探针
*
*
* 当父目录不存在时,创建目录!
*
*
* @param dirFile
*/
private static void fileProber(File dirFile) {

File parentFile = dirFile.getParentFile();
if (!parentFile.exists()) {

// 递归寻找上级目录
fileProber(parentFile);

parentFile.mkdir();
}

}

/**
* 文件解压缩
*
* @param destFile
* 目标文件
* @param zis
* ZipInputStream
* @throws Exception
*/
private static void decompressFile(File destFile, ZipInputStream zis)
throws Exception {

BufferedOutputStream bos = new BufferedOutputStream(
new FileOutputStream(destFile));

int count;
byte data[] = new byte[BUFFER];
while ((count = zis.read(data, 0, BUFFER)) != -1) {
bos.write(data, 0, count);
}

bos.close();
}

}




其实,理解了ZIP的工作原理,这些代码看起来很好懂! :D

把刚才做的压缩文件再用上述代码解开看看,测试用例如下:


/**
* 2010-4-12
*/
package org.zlex.commons.io;

import static org.junit.Assert.*;

import org.junit.Test;

/**
*
* @author 梁栋
* @version 1.0
* @since 1.0
*/
public class ZipUtilsTest {

/**
*
*/
@Test
public void test() throws Exception {
// 解压到指定目录
ZipUtils.decompress("d:\\f.txt.zip", "d:\\ff");
// 解压到当前目录
ZipUtils.decompress("d:\\fd.zip");
}

}


[b]完整代码详见附件![/b] :D

java原生的ZIP实现虽然在压缩时会因与系统字符集不符产生中文乱码,但在解压缩后,字符集即可恢复。

除了java原生的ZIP实现外,commons和ant也提供了相应的ZIP算法实现,有机会我再一一介绍! :D

相关链接:
[url=http://snowolf.iteye.com/blog/465433]Java压缩技术(一) ZLib[/url]
[url=http://snowolf.iteye.com/blog/642298]Java压缩技术(二) ZIP压缩——Java原生实现[/url]
[url=http://snowolf.iteye.com/blog/642492]Java压缩技术(三) ZIP解压缩——Java原生实现[/url]
[url=http://snowolf.iteye.com/blog/643010]Java压缩技术(四) GZIP——Java原生实现[/url]
[url=http://snowolf.iteye.com/blog/643443]Java压缩技术(五) GZIP相关——浏览器解析[/url]
[url=http://snowolf.iteye.com/blog/644591]Java压缩技术(六) BZIP2——Commons实现[/url]
[url=http://snowolf.iteye.com/blog/648652]Java压缩技术(七) TAR——Commons实现[/url]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值