由於網絡帶寬有限,所以數據文件的壓縮有利於數據在Internet上的快速傳輸,同時也節
省服務器的外存空間。
Java 1.1實現了I/O數據流與網絡數據流的單一接口,因此數據的壓縮、網絡傳輸和解
壓縮的實現比較容易,下面介紹利用ZipEntry、ZipInputStream和ZipOutputStream三個Java
類實現zip數據壓縮方式的編程方法。
zip壓縮文件結構:一個zip文件由多個entry組成,每個entry有一個唯一的名稱,entry的
數據項存儲壓縮數據。
與zip文件有關的幾個Java類
·類ZipEntry
public ZipEntry(String name);
name為指定的數據項名。
·類ZipOutputStream
ZipOutputStream實現了zip壓縮文件的寫輸出流,支持壓縮和非壓縮entry。下面是它的
幾個函數:
public ZipOutputStream(OutputStream out);
∥利用輸出流out構造一個ZIP輸出流。
public void setMethod(int method);
∥設置entry壓縮方法,缺省值為DEFLATED。
public void putNextEntry(ZipEntry newe);
∥如果當前的entry存在且處於激活狀態時,關閉它,在zip文件中寫入新的entry-newe
並將數據流定位於entry數據項的起始位置,壓縮方法為setMethod指定的方法。
·類ZipInputStream
ZipInputStream實現了zip壓縮文件的讀輸入流,支持壓縮和非壓縮entry。下面是它的
幾個函數:
public ZipInputStream(InputStream in);
∥利用輸入流in構造一個ZIP輸出流。
public ZipEntry getNextEntry();
∥返回ZIP文件中的下一個entry,並將輸出流定位在此entry數據項的起始位置。
public void closeEntry();
∥關閉當前的zip entry,並將數據流定位於下一個entry的起始位置。
程序代碼及其注釋
下列的程序實現了數據文件zip方式的壓縮和解壓縮方法。randomData()函數隨機生成
50個double數據,並放在doc字符串變量中;openFile()函數讀取ZIP壓縮文件;saveFile()函數
將隨機生成的數據存到ZIP格式的壓縮文件中。
import java.util.zip.*;
import java.awt.event.*;
import java.awt.*;
import java.lang.Math;
import java.io.*;
public class TestZip extends Frame implements ActionListener {
TextArea textarea; ∥顯示數據文件的多行文本顯示域
TextField infotip; ∥顯示數據文件未壓縮大小及壓縮大小單行文本顯示域
String doc; ∥存儲隨機生成的數據
long doczipsize = 0;∥壓縮數據文件的大小
public TestZip(){
∥生成菜單
MenuBar menubar = new MenuBar();
setMenuBar(menubar);
Menu file = new Menu("File",true);
menubar.add(file);
MenuItem neww= new MenuItem("New");
neww.addActionListener(this);
file.add(neww);
MenuItem open=new MenuItem("Open");
open.addActionListener(this);
file.add(open);
MenuItem save=new MenuItem("Save");
save.addActionListener(this);
file.add(save);
MenuItem exit=new MenuItem("Exit");
exit.addActionListener(this);
file.add(exit);
∥隨機生成的數據文件的多行文本顯示域
add("Center",textarea = new TextArea());
∥提示文本原始大小、壓縮大小的單行文本顯示域
add("South",infotip = new TextField());
}
public static void main(String args[]){
TestZip ok=new TestZip();
ok.setTitle("zip sample");
ok.setSize(600,300);
ok.show();
}
private void randomData(){
∥隨機生成50個double數據,並放在doc字符串變量中。
doc="";
for(int i=1;i<51;i++){
double rdm=Math.random()*10;
doc=doc+new Double(rdm).toString();
if(i%5 == 0) doc=doc+"n";
else doc=doc+" ";
}
doczipsize = 0;
showTextandInfo();
}
private void openFile(){
∥打開zip文件,將文件內容讀入doc字符串變量中。
FileDialog dlg=new FileDialog(this,"Open",FileDialog.LOA D);
dlg.show();
String filename=dlg.getDirectory()+dlg.getFile();
try{
∥創建一個文件實例
File f=new File(filename);
if(!f.exists()) return; ∥文件不存在,則返回
∥用文件輸入流構建ZIP壓縮輸入流
ZipInputStream zipis=new ZipInputStream(new FileInputStream(f));
zipis.getNextEntry();
∥將輸入流定位在當前entry數據項位置
DataInputStream dis=new DataInputStream(zipis);
∥用ZIP輸入流構建DataInputStream
doc=dis.readUTF();∥讀取文件內容
dis.close();∥關閉文件
doczipsize = f.length();∥獲取ZIP文件長度
showTextandInfo();∥顯示數據
}
catch(IOException ioe){
System.out.println(ioe);
}
}
private void saveFile(){
∥打開zip文件,將doc字符串變量寫入zip文件中。
FileDialog dlg=new FileDialog(this,"Save",FileDialog.SAVE);
dlg.show();
String filename=dlg.getDirectory()+dlg.getFile();
try{
∥創建一個文件實例
File f=new File(filename);
if(!f.exists()) return; ∥文件不存在,則返回
∥用文件輸出流構建ZIP壓縮輸出流
ZipOutputStream zipos=new ZipOutputStream(new FileOutputStream(f));
zipos.setMethod(ZipOutputStream.DEFLATED); ∥設置壓縮方法
zipos.putNextEntry(new ZipEntry("zip"));
∥生成一個ZIP entry,寫入文件輸出流中,並將輸出流定位於entry起始處。
DataOutputStream os=new DataOutputStream(zipos);
∥用ZIP輸出流構建DataOutputStream;
os.writeUTF(doc);∥將隨機生成的數據寫入文件中
os.close();∥關閉數據流
doczipsize = f.length();
∥獲取壓縮文件的長度
showTextandInfo();∥顯示數據
}
catch(IOException ioe){
System.out.println(ioe);
}
}
private void showTextandInfo(){
∥顯示數據文件和壓縮信息
textarea.replaceRange(doc,0,textarea.getText().length());
infotip.setText("uncompressed size: "+doc.length()+"compressed size: "+dc zipsize);
}
public void actionPerformed(ActionEvent e){
String arg = e.getActionCommand();
if ("New".equals(arg)) randomData();
else if ("Open".equals(arg)) openFile();
else if ("Save".equals(arg)) saveFile();
else if ("Exit".equals(arg)){
dispose();∥關閉窗口
System.exit(0);∥關閉程序
}
else {
System.out.println("no this command!");
}
}
}