java 文件压缩 解压_Java文件压缩与解压缩(一)

package com.cn;

import java.io.File;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.InputStream;

import java.util.Enumeration;

import java.util.zip.ZipEntry;

import java.util.zip.ZipFile;

import java.util.zip.ZipOutputStream;

//错误小结:

//1 关于file.isFile()与file.isDirectory()记忆出了偏差.

// 错误以为若是Directory一定是file,file不一定是Directory

// 更正:file和Directory是两码事.只能是file或Directory中其中一个.

//2 并不是说我们执行一句File f=new File("F:\\x.txt");

// 在本地硬盘上就生成了一个x.txt文件.而应该进行如下的

// 操作才可以.

// File f=new File("F:\\x.txt");

// if (!f.exists()) {

// f.createNewFile();

// }

// 其中f.createNewFile()表示创建了一个空文件

//

// 多数的情况下,我们执行了File f=new File("F:\\x.txt")

// 以后再利用输入流,输出流对f进行操作,比如往该x.txt文件中

// 写入hello world

//

//3 血的教训:

// zip()方法中没有关闭流zos.导致压缩后的文件解压时出错

//重要总结:

//1 关于zip和unzip的核心操作对象是每一个文件!!!

// 比如碰到一个directory,那么会去遍历里面的每一个文件,挨个对其进行压缩.

// 不要错误地理解,若是一个directory,那么会将其作为一个整体进行压缩.

//2 在JAVA中每一个压缩文件都用一个ZipEntry表示

// 所以在压缩的过程中要得到每个文件的完整路径(从最外层文件夹至文件本身)

// 用此完整路径为每个压缩文件new()一个ZipEntry

//3 所以利用zipEntry可记录原来的目录层次.解压后才保持原样

// 也可以看到在解压时利用entrys.hasMoreElements()来挨个

// 解压每个zipEntry.

// 参见解压中的代码:

// perUnzipFilePath = unzipPath + zipEntry.getName();

public class TestZipAndUnZip {

public static void main(String[] args) throws Exception {

TestZipAndUnZip test=new TestZipAndUnZip();

//压缩和解压单个文件

test.zip2("F:\\kk\\cc.txt","F:\\11.zip");

test.unZipFile2("F:\\11.zip", "F:\\test11");

//压缩和解压一个目录

test.zip2("F:\\kk","F:\\22.zip");

test.unZipFile2("F:\\22.zip", "F:\\test22");

}

/**

* 该方法将一个给定路径的文件压缩

* @param willZipPath 待压缩文件的路径

* @param zipedPath 该文件压缩后的路径

*/

public void zip2(String willZipPath, String zipedPath) {

try {

File willZipFile = new File(willZipPath);

File zipedFile = new File(zipedPath);

ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(zipedFile));

if (willZipFile.isFile()) {

fileToZip2(willZipPath, zos);

}

if (willZipFile.isDirectory()) {

dirToZip2(willZipPath, willZipFile, zos);

}

// 方法调用完成后关闭流

zos.close();

} catch (Exception e) {

}

}

/**

* @param willZipFilePath 待压缩文件的路径

* @param zos 压缩文件输出流

* 1 关于以下两句代码

* ZipEntry entry = new ZipEntry();

* zos.putNextEntry(entry);

* 把生成的ZipEntry对象加入到压缩文件中

* 之后往压缩文件中写入的内容均放在该ZipEntry对象中

*

* 2 fis.close()但是不能在此处zos.close()

* 因为该zos是上一方法传递过来的.可能在压缩目录的时候会

* 再次使用到该zos流.若此时关闭,则导致目录中的一个文件

* 被压缩

*/

public void fileToZip2(String willZipFilePath,ZipOutputStream zos){

try {

File willZipFile=new File(willZipFilePath);

ZipEntry entry = new ZipEntry(getEntryName2(willZipFilePath, willZipFile));

zos.putNextEntry(entry);

FileInputStream fis = new FileInputStream(willZipFilePath);

int len = 0;

while ((len = fis.read()) != -1){

zos.write(len);

}

fis.close();

//流关闭错误!

//zos.close();

} catch (Exception e) {

}

}

/**

* @param willZipDirctoryPath 原目录所在路径

* @param willZipedDirectory 原目录

* @param zos 压缩流

* 注意:

* 在处理空文件夹的时候

* getEntryName2(willZipDirctoryPath, willZipedDirectory)+"/"

* 中的"/"是必不可少的

*/

public void dirToZip2(String willZipDirctoryPath,File willZipedDirectory, ZipOutputStream zos) {

if (willZipedDirectory.isDirectory()) {

File[] files = willZipedDirectory.listFiles();

//处理空文件夹的情况

if (files.length==0) {

ZipEntry zipEntry=new ZipEntry

(getEntryName2(willZipDirctoryPath, willZipedDirectory)+"/");

try {

zos.putNextEntry(zipEntry);

} catch (Exception e) {

e.printStackTrace();

}

return;

}

for (int i = 0; i < files.length; i++) {

File file = files[i];

//若是文件,递归调用fileToZip()

if (file.isFile()) {

fileToZip2(file.getAbsolutePath(), zos);

}

//若是文件,递归调用dirToZip()

if (file.isDirectory()) {

dirToZip2(file.getAbsolutePath(),file, zos);

}

}

}

}

/**

* @param rawPath 需要压缩的目录或者文件的完整路径

* @param file 需要压缩的文件或者目录

* @return entryName

*

* 该方法返回EntryName,表示从最外层目录开始到该文件(目录)

* 的完整路径

* 备注:

* 1 该示例中文件均存放在某盘下,如E:\所以rawPath.substring(3);

* 2 注释中"@param file 需要压缩的文件或者目录".其实绝大多数情况下

* 都是文件,只有一种情况是目录,就是空文件夹的情况.

*/

public String getEntryName2(String rawPath,File file){

try {

String rawDir=rawPath.substring(3);

int rawDirIndex=file.getAbsolutePath().indexOf(rawDir);

String entryName=file.getAbsolutePath().substring(rawDirIndex);

return entryName;

} catch (Exception e) {

}

return null;

}

/**

* @param zipedFilePath 原压缩文件的路径

* @param unzipPath 文件解压后的路径

* 对于文件或者目录操作的小梳理:

* 1 对于目录应该先执行file.mkdir(s)()才可以

* 往里面其下存入文件.比如:

* File f=new File("F:\\test\\x.txt");

if (!f.exists()) {

f.createNewFile();

}

这当然要报错,因为x.txt的所在目录还不存在!!

所以应该改正为:

File f=new File("F:\\test\\x.txt");

f.getParentFile().mkdirs();

if (!f.exists()) {

f.createNewFile();

}

2同样的道理

File f=new File("F:\\test\\x.txt");

if (f.isFile()) {

System.out.println("true");

}else{

System.out.println("false");

}

结果为false

3类似的问题

File f=new File("F:\\x.txt");

if (f.isFile()) {

System.out.println("true");

}else{

System.out.println("false");

}

结果为false

因为只是new了一个File,并没有创建!!!

File f=new File("F:\\x.txt");

f.createNewFile();

if (f.isFile()) {

System.out.println("true");

}else{

System.out.println("false");

}

此时为true

此处:

if (zipEntry.isDirectory()) {

new File(perUnzipFilePath).mkdirs();

} else {

new File(perUnzipFilePath).getParentFile().mkdirs();

}

已经建立了每个文件夹.

然后才开始对每个空文件夹和每个文件进行流操作.

和上面的道理一致,刚开始犯了个错误,没有采用

else {

fos = new FileOutputStream(perUnzipFile);

is = zipFile.getInputStream(zipEntry);

while ((len = is.read(buffer)) != -1) {

fos.write(buffer, 0, len);

}

}

而是用if(perUnzipFile.isFile()){}来操作,这当然错了.

因为该perUnzipFile没有执行perUnzipFile.createNewFile();

所以它还不是File.类似的情况,多采用流操作来进行读写.

所以总结两种与File有关的操作:

1 File f=new File("");

f.createNewFile();

然后对f操作

2 file f=new File("");

然后用输入输出流进行流操作

举例子:

File f=new File("F:\\2221x.txt");

FileOutputStream fos=new FileOutputStream(f);

String string="hello";

byte []b=string.getBytes();

fos.write(b, 0, b.length);

该例子是正确的.

疑问:没有执行 f.createNewFile()为什么不报错.因为输出流

FileOutputStream已经帮我们做了该工作了.

修改例子即可知:

File f=new File("F:\\2221x.txt");

if (f.isFile()) {

System.out.println("true1");

} else {

System.out.println("false1");

}

FileOutputStream fos=new FileOutputStream(f);

if (f.isFile()) {

System.out.println("true2");

} else {

System.out.println("false2");

}

String string="hello";

byte []b=string.getBytes();

fos.write(b, 0, b.length);

输出false1,true2

这就得到了验证.

*/

public void unZipFile2(String zipedFilePath, String unzipPath) {

FileOutputStream fos=null;

InputStream is=null;

ZipEntry zipEntry=null;

String perUnzipFilePath=null;

if (!unzipPath.endsWith(File.separator)) {

unzipPath+=File.separator;

}

try {

ZipFile zipFile=new ZipFile(zipedFilePath);

Enumeration entries=zipFile.entries();

byte [] buffer=new byte[1024*8];

int len=0;

while (entries.hasMoreElements()) {

zipEntry = (ZipEntry) entries.nextElement();

perUnzipFilePath = unzipPath + zipEntry.getName();

//1建立每个文件夹

if (zipEntry.isDirectory()) {

//处理空文件夹的情况

//创建空目录

new File(perUnzipFilePath).mkdirs();

} else {

//为每个文件创立其所在目录

new File(perUnzipFilePath).getParentFile().mkdirs();

}

//2在用流操作处理每个文件夹中的文件

// 2.1if中的操作只是针对空目录而进行的.所以if中的代码可以

// 注释掉,无实际意义.

// 2.2else中的操作是对于每个具体的文件而进行的流操作

File perUnzipFile = new File(perUnzipFilePath);

if (perUnzipFile.isDirectory()) {

File[] files = perUnzipFile.listFiles();

for (int i = 0; i < files.length; i++) {

File file = files[i];

fos = new FileOutputStream(file);

is = zipFile.getInputStream(zipEntry);

while ((len = is.read(buffer)) != -1) {

fos.write(buffer, 0, len);

}

}

} else {

fos = new FileOutputStream(perUnzipFile);

is = zipFile.getInputStream(zipEntry);

while ((len = is.read(buffer)) != -1) {

fos.write(buffer, 0, len);

}

}

}

if (fos!=null) {

fos.close();

}

if (is!=null) {

is.close();

}

} catch (Exception e) {

e.printStackTrace();

}

}

}

ps:

Java文件压缩与解压缩(一),最好.

Java文件压缩与解压缩(三),其次.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值