ZIP4j 壓縮與解壓的實(shí)例詳解
ZIP4j 壓縮與解壓的實(shí)例詳解
使用的jar包:zip4j_1.3.2.jar
基本功能:
針對(duì)ZIP壓縮文件創(chuàng)建、添加、分卷、更新和移除文件
(讀寫有密碼保護(hù)的Zip文件)
(支持AES 128/256算法加密)
(支持標(biāo)準(zhǔn)Zip算法加密)
(支持zip64格式)
(支持Store(僅打包,默認(rèn)不壓縮,不過可以手動(dòng)設(shè)置大小)和Deflate壓縮方法
(針對(duì)分塊zip文件創(chuàng)建和抽出文件)
(支持編碼)
(進(jìn)度監(jiān)控)
壓縮方式(3種):
static final int COMP_STORE = 0;(僅打包,不壓縮) (對(duì)應(yīng)好壓的存儲(chǔ)) static final int COMP_DEFLATE = 8;(默認(rèn)) (對(duì)應(yīng)好壓的標(biāo)準(zhǔn)) static final int COMP_AES_ENC = 99;
壓縮級(jí)別有5種:(默認(rèn)0不壓縮)級(jí)別跟好壓軟件是對(duì)應(yīng)的;
static final int DEFLATE_LEVEL_FASTEST = 1; static final int DEFLATE_LEVEL_FAST = 3; static final int DEFLATE_LEVEL_NORMAL = 5; static final int DEFLATE_LEVEL_MAXIMUM = 7; static final int DEFLATE_LEVEL_ULTRA = 9;
加密方式:
static final int ENC_NO_ENCRYPTION = -1;(默認(rèn),沒有加密方法,如果采用此字段,會(huì)報(bào)錯(cuò)”沒有提供加密算法”) static final int ENC_METHOD_STANDARD = 0; static final int ENC_METHOD_AES = 99;
AES Key Strength:
(默認(rèn)-1,也就是ENC_NO_ENCRYPTION) static final int AES_STRENGTH_128 = 0x01; static final int AES_STRENGTH_192 = 0x02; static final int AES_STRENGTH_256 = 0x03;
從構(gòu)造方法可以默認(rèn)情況:
compressionMethod = Zip4jConstants.COMP_DEFLATE; encryptFiles = false;//不設(shè)密碼 readHiddenFiles = true;//可見 encryptionMethod = Zip4jConstants.ENC_NO_ENCRYPTION;//加密方式不加密 aesKeyStrength = -1;// includeRootFolder = true;// timeZone = TimeZone.getDefault();//
發(fā)現(xiàn)的現(xiàn)象:
在采取默認(rèn)壓縮時(shí):
1.如果此壓縮文件已經(jīng)存在,那么壓縮后,相同的文件會(huì)替換(有密碼,密碼被替換),原來不同的文件會(huì)繼續(xù)保留,而且文件的時(shí)間還是第一次壓縮的時(shí)間;如果想完全覆蓋,那么需要判斷此壓縮文件存不存在,存在就刪除;
2.假如a文件加密生成了a.zip,此時(shí)如果再把其他的文件b也加密,然后生成同樣的a.zip,那么a.zip里面的文件a,b將會(huì)有各自的密碼。需要分別輸入對(duì)應(yīng)密碼解壓,無形實(shí)現(xiàn)了對(duì)單個(gè)文件的單個(gè)加密,但是這樣解壓可能會(huì)損壞文件(個(gè)人不建議這樣做)
3.如果不設(shè)置壓縮級(jí)別,默認(rèn)級(jí)別為0(不壓縮),這樣生成的zip包跟原來文件的大小差不多,另外如果壓縮方式設(shè)置成了Zip4jConstants.COMP_STORE(0)那么也是僅僅打個(gè)包;
4.如果設(shè)置了中文密碼,用好壓解壓會(huì)提示密碼錯(cuò)誤(無法解壓),用ZIP4j解壓的話,正常;也就說,對(duì)于中文密碼,解壓軟件與zip4j是不能相互解壓的,原因還不清楚,請(qǐng)大神說說;
下面先看壓縮文件夾: (author:余紹陽(yáng),qq:279907389)
//壓縮單個(gè)文件
@Test
public void dozip1() throws ZipException {
ZipFile zip=new ZipFile("D://壓縮//test.zip");
//需要判斷壓縮父路徑是否存在
File file=zip.getFile();
if (!file.getParentFile().exists()) {
file.getParentFile().mkdirs();
}
//設(shè)置參數(shù)
ZipParameters para=new ZipParameters();
//設(shè)置壓縮方式,默認(rèn)是COMP_DEFLATE
para.setCompressionMethod(Zip4jConstants.COMP_DEFLATE);
//設(shè)置壓縮級(jí)別,默認(rèn)為0(即不壓縮)
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL);
//設(shè)置壓縮密碼(至少3步)
//設(shè)置加密文件
para.setEncryptFiles(true);
//設(shè)置加密方式(必須要有加密算法)
para.setEncryptionMethod(Zip4jConstants.ENC_METHOD_STANDARD);
para.setPassword("test111");
File srcFile=new File("src//com//學(xué)習(xí)路線.png");
zip.addFile(srcFile, para);
System.out.println("壓縮成功!");
}
//壓縮多個(gè)文件
@Test
public void dozip2() throws ZipException {
ZipFile zip=new ZipFile("D://壓縮//test.zip");
//要緊跟設(shè)置編碼
zip.setFileNameCharset("GBK");
ArrayList<File> list=new ArrayList<>();
list.add(new File("src//com//壓縮解壓//abc.txt"));
list.add(new File("src//com//壓縮解壓//abcUTF-8.txt"));
list.add(new File("src//com//壓縮解壓//解壓之后.txt"));
ZipParameters para=new ZipParameters();
para.setCompressionMethod(Zip4jConstants.COMP_DEFLATE);
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL);
//設(shè)置密碼:
para.setEncryptFiles(true);
//設(shè)置AES加密方式
para.setEncryptionMethod(Zip4jConstants.ENC_METHOD_AES);
//必須設(shè)置長(zhǎng)度
para.setAesKeyStrength(Zip4jConstants.AES_STRENGTH_256);
para.setPassword("111");
zip.addFiles(list, para);
System.out.println("壓縮成功!");
}
//壓縮文件夾
@Test
public void dozip3() throws ZipException {
ZipFile zip=new ZipFile("D:\\壓縮\\test.zip");
File file=zip.getFile();
if (!file.getParentFile().exists()) {
file.getParentFile().mkdirs();
}
//為了不被原有文件干擾,保證每次重新生成
if (file.exists()) {
file.delete();
}
ZipParameters para=new ZipParameters();
//設(shè)置壓縮級(jí)別,壓縮方法默認(rèn)
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL);
zip.addFolder("E:\\測(cè)試壓縮", para);
System.out.println("壓縮完成!");
解壓
1.如果是用軟件壓縮(GBK)的,用zip4j解壓時(shí)需要設(shè)置”GBK”
注意設(shè)置要緊接ZipFile設(shè)置,否則編碼設(shè)置會(huì)失效;
ZipFile zipFile = new ZipFile(“D:\壓縮\test.zip”); zip.setFileNameCharset(“GBK”);
另外:在解壓之后,如果再次解壓,如果有新的文件會(huì)添加進(jìn)去,相同的文件會(huì)覆蓋;不同編碼格式的會(huì)保留;
2.zip4j是支持中文密碼的,但是如果是好壓或者winRAR壓縮并且設(shè)置中文密碼,則使用zip4j解壓會(huì)提示密碼錯(cuò)誤;
也就是說,用好壓設(shè)置中文密碼,只能用好壓或者winRAR等軟件解壓;而使用zip4j設(shè)置的中文密碼,好壓或winRAR也不能解壓,只能用自帶的zip4j解壓; 原因不清
3.密碼2種設(shè)置方式:
a.para.setEncryptFiles(true);
//沒有使用AES,那么就不需要AesKeyStrength para.setEncryptionMethod(Zip4jConstants.ENC_METHOD_STANDARD) para.setPassword(“111”);
b.para.setEncryptFiles(true);
para.setEncryptionMethod(Zip4jConstants.ENC_METHOD_AES); //如果設(shè)置AES加密,那么必須指定AesKeyStrength parameters.setAesKeyStrength(Zip4jConstants.AES_STRENGTH_256); para.setPassword(“111”);
c.如果para.setEncryptFiles(false);那么密碼將會(huì)失效
// 解壓方法1
public void Unzip4j(String zipFile) throws ZipException {
long startTime = System.currentTimeMillis();
//第一時(shí)間設(shè)置編碼格式
zip.setFileNameCharset("GBK");
//用自帶的方法檢測(cè)一下zip文件是否合法,包括文件是否存在、是否為zip文件、是否被損壞等
if (!zip.isValidZipFile()) {
throw new ZipException("文件不合法或不存在");
}
checkEncrypted(zip);
// 跟java自帶相比,這里文件路徑會(huì)自動(dòng)生成,不用判斷
zip.extractAll("D:\\壓縮\\test1");
System.out.println("解壓成功!");
long endTime = System.currentTimeMillis();
System.out.println("耗時(shí):" + (endTime - startTime) + "ms");
}
// 解壓方法2
public void Unzip4j1(String zipFile) throws ZipException {
long startTime = System.currentTimeMillis();
ZipFile zipFile2 = new ZipFile(zipFile);
//設(shè)置編碼格式
zipFile2.setFileNameCharset("GBK");
if (!zipFile2.isValidZipFile()) {
throw new ZipException("文件不合法或不存在");
}
//檢查是否需要密碼
checkEncrypted(zipFile2);
List<FileHeader> fileHeaderList = zipFile2.getFileHeaders();
for (int i = 0; i < fileHeaderList.size(); i++) {
FileHeader fileHeader = fileHeaderList.get(i);
zipFile2.extractFile(fileHeader, "D:\\壓縮\\test");
}
System.out.println("解壓成功!");
long endTime = System.currentTimeMillis();
System.out.println("耗時(shí):" + (endTime - startTime) + "ms");
}
//檢測(cè)密碼
private void checkEncrypted(ZipFile zip) throws ZipException {
Scanner in=new Scanner(System.in);
if (zip.isEncrypted()) {
System.out.println("文件"+zip.getFile().getName()+"有密碼!");
System.out.println("請(qǐng)輸入密碼:");
zip.setPassword(in.next().trim());
}
in.close();
}
其他功能
1.添加:是往zip文件添加非zip文件
在添加文件時(shí),發(fā)現(xiàn)只能添加一次,再次添加報(bào)錯(cuò):Expected central directory entry not found (#1),原因還不知道,55
另外,如果要添加的zip文件不存在,那么將會(huì)創(chuàng)建一個(gè)zip文件,再添加文件;如果要添加的a.txt文件不存在,則報(bào)錯(cuò)”沒有找到文件”
2.分卷:是壓縮的時(shí)候進(jìn)行,可針對(duì)帶個(gè)文件或者文件夾 ,分卷后每份壓縮文件大小至少64k(65536byte) 如果在對(duì)文件(文件夾)分卷壓縮前,其已經(jīng)壓縮了,那么將會(huì)分卷報(bào)錯(cuò)! 那么如何解壓呢?對(duì)任意一個(gè)分卷壓縮文件解壓即可全部解壓;
3.預(yù)覽壓縮文件信息
4.刪除功能在下一篇展示,這個(gè)在zip文件刪除指定文件夾有一種非常巧妙的方法(非常規(guī)思路),不需要遞歸;
代碼如下:
public static void addFile(String srczipFile, String addfile)
throws Exception {
ZipFile zip = new ZipFile(srczipFile);
ZipParameters para = new ZipParameters();
// 封裝待添加文件
FileInputStream fis = new FileInputStream(addfile);
String fileName = new File(addfile).getName();
System.out.println(fileName);
// 這里可以設(shè)置文件夾,會(huì)自動(dòng)在壓縮文件中創(chuàng)建這些文件夾
// String Folder="1\\2\\";
// para.setFileNameInZip(Folder+fileName);
para.setFileNameInZip(fileName);
// 資源擴(kuò)展
para.setSourceExternalStream(true);
// 添加文件
zip.addStream(fis, para);
fis.close();
System.out.println("添加成功!");
}
// 分卷,將幾個(gè)文件進(jìn)行分卷壓縮,一個(gè)單位64k
public static void zipTOSplit(String destZipFile, int unit)
throws ZipException {
ZipFile zip = new ZipFile(destZipFile);
ArrayList<File> list = new ArrayList<File>();
list.add(new File("src//com//壓縮解壓//abc.txt"));
list.add(new File("src//com//壓縮解壓//abcUTF-8.txt"));
list.add(new File("src//com//cbzk//壓縮解壓//解壓之后.txt"));
ZipParameters para = new ZipParameters();
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_FAST);
// 創(chuàng)建分卷
zip.createZipFile(list, para, true, unit * 65536);
System.out.println("分卷成功!");
}
// 將文件夾進(jìn)行分卷壓縮
public static void zipTOSplit(String destZipFile, String srcFolder, int unit)
throws ZipException {
ZipFile zip = new ZipFile(destZipFile);
ZipParameters para = new ZipParameters();
// 默認(rèn)COMP_DEFLATE
para.setCompressionMethod(Zip4jConstants.COMP_DEFLATE);
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL);
// 分卷壓縮文件夾到指定壓縮位置
zip.createZipFileFromFolder(srcFolder, para, true, unit * 65536);
//粗略的算一下分成多少份,獲取的大小比實(shí)際的大點(diǎn)(一般是準(zhǔn)確的)
int partsize=(int)zipInfo(destZipFile)/(unit*64); //65536byte=64kb
System.out.println("分割成功!總共分割成了"+(partsize+1)+"個(gè)文件!");
}
// 預(yù)覽壓縮文件信息
public static double zipInfo(String zipFile) throws ZipException {
ZipFile zip = new ZipFile(zipFile);
zip.setFileNameCharset("GBK");
List<FileHeader> list = zip.getFileHeaders();
long zipCompressedSize=0;
for(FileHeader head:list){
zipCompressedSize+=head.getCompressedSize();
// System.out.println(zipFile+"文件相關(guān)信息如下:");
// System.out.println("Name: "+head.getFileName());
// System.out.println("Compressed Size:"+(head.getCompressedSize()/1.0/1024)+"kb");
// System.out.println("Uncompressed Size:"+(head.getUncompressedSize()/1.0/1024)+"kb");
// System.out.println("CRC32:"+head.getCrc32());
// System.out.println("*************************************");
}
double size=zipCompressedSize/1.0/1024;//轉(zhuǎn)換為kb
return size;
}
如有疑問請(qǐng)留言或者到本站社區(qū)交流討論,感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
String在棧中,StringBuffer在堆中!所以String是不可變的,數(shù)據(jù)是共享的。StringBuffer都是獨(dú)占的,是可變的(因?yàn)槊看味际莿?chuàng)建新的對(duì)象?。?/div> 2015-11-11
Java ThreadLocal的設(shè)計(jì)理念與作用
這篇文章主要介紹了Java ThreadLocal的設(shè)計(jì)理念與作用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
ExecutorService實(shí)現(xiàn)獲取線程返回值
這篇文章主要介紹了ExecutorService實(shí)現(xiàn)獲取線程返回值,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08最新評(píng)論

