当前位置: 移动技术网 > IT编程>开发语言>Java > Java实现文件压缩与解压的示例[zip格式,gzip格式]

Java实现文件压缩与解压的示例[zip格式,gzip格式]

2019年07月22日  | 移动技术网IT编程  | 我要评论
java实现zip的解压与压缩功能基本都是使用了java的多肽和递归技术,可以对单个文件和任意级联文件夹进行压缩和解压,对于一些初学者来说是个很不错的实例。 zip扮演着

java实现zip的解压与压缩功能基本都是使用了java的多肽和递归技术,可以对单个文件和任意级联文件夹进行压缩和解压,对于一些初学者来说是个很不错的实例。

zip扮演着归档和压缩两个角色;gzip并不将文件归档,仅只是对单个文件进行压缩,所以,在unix平台上,命令tar通常用来创建一个档案文件,然后命令gzip来将档案文件压缩。

java i/o类库还收录了一些能读写压缩格式流的类。要想提供压缩功能,只要把它们包在已有的i/o类的外面就行了。这些类不是reader和writer,而是inputstream和outstreamput的子类。这是因为压缩算法是针对byte而不是字符的。
相关类与接口:

checksum 接口:被类adler32和crc32实现的接口

adler32 :使用alder32算法来计算checksum数目

crc32 :使用crc32算法来计算checksum数目

checkedinputstream :inputstream派生类,可得到输入流的校验和checksum,用于校验数据的完整性

checkedoutputstream :outputstream派生类,可得到输出流的校验和checksum, 用于校验数据的完整性

deflateroutputstream :压缩类的基类。

zipoutputstream :deflateroutputstream的一个子类,把数据压缩成zip文件格式。

gzipoutputstream :deflateroutputstream的一个子类,把数据压缩成gzip文件格式

inflaterinputstream :解压缩类的基类

zipinputstream :inflaterinputstream的一个子类,能解压缩zip格式的数据

gzipinputstream :inflaterinputstream的一个子类,能解压缩zip格式的数据

zipentry 类:表示 zip 文件条目

zipfile 类:此类用于从 zip 文件读取条目

使用zip对多个文件进行压缩与解压

java对zip格式类库支持得比较全面,得用它可以把多个文件压缩成一个压缩包。这个类库使用的是标准zip格式,所以能与很多的压缩工具兼容。

zipoutputstream类有设置压缩方法以及在压缩方式下使用的压缩级别,zipoutputstream.setmethod(int method)设置用于条目的默认压缩方法。只要没有为单个 zip 文件条目指定压缩方法,就使用zipoutputstream所设置的压缩方法来存储,默认值为 zipoutputstream.deflated(表示进行压缩存储),还可以设置成stored(表示仅打包归档存储)。zipoutputstream在设置了压缩方法为deflated后,我们还可以进一步使用setlevel(int level)方法来设置压缩级别,压缩级别值为0-9共10个级别(值越大,表示压缩越利害),默认为deflater.default_compression=-1。当然我们也可以通过条目zipentry的setmethod方法为单个条件设置压缩方法。

类zipentry描述了存储在zip文件中的压缩文件。类中包含有多种方法可以用来设置和获得zip条目的信息。类zipentry是被zipfile[zipfile.getinputstream(zipentry entry)]和zipinputstream使用来读取zip文件,zipoutputstream来写入zip文件的。有以下这些有用的方法:getname()返回条目名称、isdirectory()如果为目录条目,则返回 true(目录条目定义为其名称以 ‘/' 结尾的条目)、setmethod(int method) 设置条目的压缩方法,可以为 zipoutputstream.stored 或 zipoutputstream .deflated。

下面实例我们使用了apache的zip工具包(所在包为ant.jar ),因为java类型自带的不支持中文路径,不过两者使用的方式是一样的,只是apache压缩工具多了设置编码方式的接口,其他基本上是一样的。另外,如果使用org.apache.tools.zip.zipoutputstream来压缩的话,我们只能使用org.apache.tools.zip.zipentry来解压,而不能使用java.util.zip.zipinputstream来解压读取了,当然apache并未提供zipinputstream类。

文件压缩:

package gizaction;
import java.io.*;
import java.util.zip.*;
/**
 * @author dana·li
 * <p>
 * 程序实现了zip压缩[compression]
 * <p>
 * 大致功能包括用了多态,递归等java核心技术,可以对单个文件和任意级联文件夹进行压缩和解压。 需在代码中自定义源输入路径和目标输出路径。
 * <p>
 * 在本段代码中,实现的是压缩部分
 */
public class zipcompressing {
 private int k = 1; // 定义递归次数变量

 private void zip(string zipfilename, file inputfile) throws exception {
  system.out.println("压缩中...");
  zipoutputstream out = new zipoutputstream(new fileoutputstream(zipfilename));
  bufferedoutputstream bo = new bufferedoutputstream(out);
  zip(out, inputfile, inputfile.getname(), bo);
  bo.close();
  out.close(); // 输出流关闭
  system.out.println("压缩完成");
 }
 private void zip(zipoutputstream out, file f, string base,
   bufferedoutputstream bo) throws exception { // 方法重载
  if (f.isdirectory()){
   file[] fl = f.listfiles();
   if (fl.length == 0){
    out.putnextentry(new zipentry(base + "/")); // 创建zip压缩进入点base
    system.out.println(base + "/");
   }
   for (int i = 0; i < fl.length; i++) {
    zip(out, fl[i], base + "/" + fl[i].getname(), bo); // 递归遍历子文件夹
   }
   system.out.println("第" + k + "次递归");
   k++;
  } else {
   out.putnextentry(new zipentry(base)); // 创建zip压缩进入点base
   system.out.println(base);
   fileinputstream in = new fileinputstream(f);
   bufferedinputstream bi = new bufferedinputstream(in);
   int b;
   while ((b = bi.read()) != -1) {
    bo.write(b); // 将字节流写入当前zip目录
   }
   bi.close();
   in.close(); // 输入流关闭
  }
 }
 /**
  * 测试
  * @param args
  */
 public static void main(string[] args) {
  zipcompressing book = new zipcompressing();
  try {
   book.zip("f:\\ziptest.zip",new file("f:\\ziptest"));
  } catch (exception e) {
   e.printstacktrace();
  }
 }
}

文件解压:

package gizaction;
import java.io.bufferedinputstream;
import java.io.bufferedoutputstream;
import java.io.file;
import java.io.fileinputstream;
import java.io.filenotfoundexception;
import java.io.fileoutputstream;
import java.io.ioexception;
import java.util.zip.zipentry;
import java.util.zip.zipinputstream;
/**
 * @author dana·li
 * <p>
 * 程序实现了zip解压[decompression]
 * <p>
 * 大致功能包括用了多态,递归等java核心技术,可以对单个文件和任意级联文件夹进行压缩和解压。 需在代码中自定义源输入路径和目标输出路径。
 * <p>
 * 在本段代码中,实现的是解压部分;
 */ 
public class zipdecompressing { 

 public static void main(string[] args) { 
  // todo auto-generated method stub 
  long starttime=system.currenttimemillis(); 
  try { 
   zipinputstream zin=new zipinputstream(new fileinputstream( 
     "f:\\ziptest.zip"));//输入源zip路径 
   bufferedinputstream bin=new bufferedinputstream(zin); 
   string parent="f:\\ziptest\\"; //输出路径(文件夹目录) 
   file fout=null; 
   zipentry entry; 
   try { 
    while((entry = zin.getnextentry())!=null && !entry.isdirectory()){ 
     fout=new file(parent,entry.getname()); 
     if(!fout.exists()){ 
      (new file(fout.getparent())).mkdirs(); 
     } 
     fileoutputstream out=new fileoutputstream(fout); 
     bufferedoutputstream bout=new bufferedoutputstream(out); 
     int b; 
     while((b=bin.read())!=-1){ 
      bout.write(b); 
     } 
     bout.close(); 
     out.close(); 
     system.out.println(fout+"解压成功");  
    } 
    bin.close(); 
    zin.close(); 
   } catch (ioexception e) { 
    e.printstacktrace(); 
   } 
  } catch (filenotfoundexception e) { 
   e.printstacktrace(); 
  } 
  long endtime=system.currenttimemillis(); 
  system.out.println("耗费时间: "+(endtime-starttime)+" ms"); 
 } 

}

用gzip进行对单个文件压缩

gzip的接口比较简单,因此如果你只需对一个流进行压缩的话,可以使用它。当然它可以压缩字符流,与可以压缩字节流,下面是一个对gbk编码格式的文本文件进行压缩的。

压缩类的用法非常简单;只要用gzipoutputstream 或zipoutputstream把输出流包起来,再用gzipinputstream 或zipinputstream把输入流包起来就行了。剩下的都是些普通的i/o操作。

import java.io.bufferedoutputstream;
import java.io.bufferedreader;
import java.io.fileinputstream;
import java.io.fileoutputstream;
import java.io.ioexception;
import java.io.inputstreamreader;
import java.util.zip.gzipinputstream;
import java.util.zip.gzipoutputstream;
public class gzipcompress {
 public static void main(string[] args) throws ioexception {
  //做准备压缩一个字符文件,注,这里的字符文件要是gbk编码方式的
  bufferedreader in = new bufferedreader(new inputstreamreader(new fileinputstream(
    "e:/tmp/source.txt"), "gbk"));
  //使用gzipoutputstream包装outputstream流,使其具体压缩特性,最后会生成test.txt.gz压缩包
  //并且里面有一个名为test.txt的文件
  bufferedoutputstream out = new bufferedoutputstream(new gzipoutputstream(
    new fileoutputstream("test.txt.gz")));
  system.out.println("开始写压缩文件...");
  int c;
  while ((c = in.read()) != -1) {

   /* 
    * 注,这里是压缩一个字符文件,前面是以字符流来读的,不能直接存入c,因为c已是unicode
    * 码,这样会丢掉信息的(当然本身编码格式就不对),所以这里要以gbk来解后再存入。
    */
   out.write(string.valueof((char) c).getbytes("gbk"));
  }
  in.close();
  out.close();
  system.out.println("开始读压缩文件...");
  //使用gzipinputstream包装inputstream流,使其具有解压特性
  bufferedreader in2 = new bufferedreader(new inputstreamreader(
    new gzipinputstream(new fileinputstream("test.txt.gz")), "gbk"));
  string s;
  //读取压缩文件里的内容
  while ((s = in2.readline()) != null) {
   system.out.println(s);
  }
  in2.close();
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持移动技术网。

如您对本文有疑问或者有任何想说的,请点击进行留言回复,万千网友为您解惑!

相关文章:

验证码:
移动技术网