热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

ZIP4j压缩与解压

使用的jar包:zip4j_1.3.2.jar基本功能:针对ZIP压缩文件创建、添加、分卷、更新和移除文件(读写有密码保护的Zip文件)(支持AES128256算法加密)

使用的jar包:zip4j_1.3.2.jar
基本功能:
针对ZIP压缩文件创建、添加、分卷、更新和移除文件
(读写有密码保护的Zip文件)
(支持AES 128/256算法加密)
(支持标准Zip算法加密)
(支持zip64格式)
(支持Store(仅打包,默认不压缩,不过可以手动设置大小)和Deflate压缩方法
(针对分块zip文件创建和抽出文件)
(支持编码)
(进度监控)
压缩方式(3种):
static final int COMP_STORE = 0;(仅打包,不压缩) (对应好压的存储)
static final int COMP_DEFLATE = 8;(默认) (对应好压的标准)
static final int COMP_AES_ENC = 99;

压缩级别有5种:(默认0不压缩)级别跟好压软件是对应的;
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;(默认,没有加密方法,如果采用此字段,会报错”没有提供加密算法”)
static final int ENC_METHOD_STANDARD = 0;
static final int ENC_METHOD_AES = 99;
AES Key Strength:
(默认-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;

从构造方法可以默认情况:
compressiOnMethod= Zip4jConstants.COMP_DEFLATE;
encryptFiles = false;//不设密码
readHiddenFiles = true;//可见
encryptiOnMethod= Zip4jConstants.ENC_NO_ENCRYPTION;//加密方式不加密
aesKeyStrength = -1;//
includeRootFolder = true;//
timeZOne= TimeZone.getDefault();//

发现的现象:
在采取默认压缩时:
1.如果此压缩文件已经存在,那么压缩后,相同的文件会替换(有密码,密码被替换),原来不同的文件会继续保留,而且文件的时间还是第一次压缩的时间;如果想完全覆盖,那么需要判断此压缩文件存不存在,存在就删除;
2.假如a文件加密生成了a.zip,此时如果再把其他的文件b也加密,然后生成同样的a.zip,那么a.zip里面的文件a,b将会有各自的密码。需要分别输入对应密码解压,无形实现了对单个文件的单个加密,但是这样解压可能会损坏文件(个人不建议这样做)
3.如果不设置压缩级别,默认级别为0(不压缩),这样生成的zip包跟原来文件的大小差不多,另外如果压缩方式设置成了Zip4jConstants.COMP_STORE(0)那么也是仅仅打个包;
4.如果设置了中文密码,用好压解压会提示密码错误(无法解压),用ZIP4j解压的话,正常;也就说,对于中文密码,解压软件与zip4j是不能相互解压的,原因还不清楚,请大神说说;
下面先看压缩文件夹:
(author:余绍阳,qq:279907389)

 //压缩单个文件
@Test
public void dozip1() throws ZipException {
ZipFile zip=new ZipFile("D://压缩//test.zip");
//需要判断压缩父路径是否存在
File file=zip.getFile();
if (!file.getParentFile().exists()) {
file.getParentFile().mkdirs();
}

//设置参数
ZipParameters para=new ZipParameters();
//设置压缩方式,默认是COMP_DEFLATE
para.setCompressionMethod(Zip4jConstants.COMP_DEFLATE);
//设置压缩级别,默认为0(即不压缩)
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL);
//设置压缩密码(至少3步)
//设置加密文件
para.setEncryptFiles(true);
//设置加密方式(必须要有加密算法)
para.setEncryptionMethod(Zip4jConstants.ENC_METHOD_STANDARD);
para.setPassword("test111");
File srcFile=new File("src//com//学习路线.png");
zip.addFile(srcFile, para);
System.out.println("压缩成功!");
}
//压缩多个文件
@Test
public void dozip2() throws ZipException {
ZipFile zip=new ZipFile("D://压缩//test.zip");
//要紧跟设置编码
zip.setFileNameCharset("GBK");
ArrayList 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);
//设置密码:
para.setEncryptFiles(true);
//设置AES加密方式
para.setEncryptionMethod(Zip4jConstants.ENC_METHOD_AES);
//必须设置长度
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();
//设置压缩级别,压缩方法默认
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL);
zip.addFolder("E:\\测试压缩", para);
System.out.println("压缩完成!");

解压

1.如果是用软件压缩(GBK)的,用zip4j解压时需要设置”GBK”
注意设置要紧接ZipFile设置,否则编码设置会失效;
ZipFile zipFile = new ZipFile(“D:\压缩\test.zip”);
zip.setFileNameCharset(“GBK”);

另外:在解压之后,如果再次解压,如果有新的文件会添加进去,相同的文件会覆盖;不同编码格式的会保留;
2.zip4j是支持中文密码的,但是如果是好压或者winRAR压缩并且设置中文密码,则使用zip4j解压会提示密码错误;
也就是说,用好压设置中文密码,只能用好压或者winRAR等软件解压;而使用zip4j设置的中文密码,好压或winRAR也不能解压,只能用自带的zip4j解压; 原因不清
3.密码2种设置方式:
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);
//如果设置AES加密,那么必须指定AesKeyStrength
parameters.setAesKeyStrength(Zip4jConstants.AES_STRENGTH_256);
para.setPassword(“111”);

c.如果para.setEncryptFiles(false);那么密码将会失效

 // 解压方法1
public void Unzip4j(String zipFile) throws ZipException {
long startTime = System.currentTimeMillis();
//第一时间设置编码格式
zip.setFileNameCharset("GBK");
//用自带的方法检测一下zip文件是否合法,包括文件是否存在、是否为zip文件、是否被损坏等
if (!zip.isValidZipFile()) {
throw new ZipException("文件不合法或不存在");
}
checkEncrypted(zip);
// 跟java自带相比,这里文件路径会自动生成,不用判断
zip.extractAll("D:\\压缩\\test1");
System.out.println("解压成功!");
long endTime = System.currentTimeMillis();
System.out.println("耗时:" + (endTime - startTime) + "ms");
}

// 解压方法2
public void Unzip4j1(String zipFile) throws ZipException {
long startTime = System.currentTimeMillis();
ZipFile zipFile2 = new ZipFile(zipFile);
//设置编码格式
zipFile2.setFileNameCharset("GBK");
if (!zipFile2.isValidZipFile()) {
throw new ZipException("文件不合法或不存在");
}
//检查是否需要密码
checkEncrypted(zipFile2);
List fileHeaderList = zipFile2.getFileHeaders();
for (int i = 0; i FileHeader fileHeader = fileHeaderList.get(i);
zipFile2.extractFile(fileHeader, "D:\\压缩\\test");
}
System.out.println("解压成功!");
long endTime = System.currentTimeMillis();
System.out.println("耗时:" + (endTime - startTime) + "ms");
}

//检测密码
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("请输入密码:");
zip.setPassword(in.next().trim());
}
in.close();
}

其他功能

1.添加:是往zip文件添加非zip文件
在添加文件时,发现只能添加一次,再次添加报错:Expected central directory entry not found (#1),原因还不知道,55
另外,如果要添加的zip文件不存在,那么将会创建一个zip文件,再添加文件;如果要添加的a.txt文件不存在,则报错”没有找到文件”
2.分卷:是压缩的时候进行,可针对带个文件或者文件夹
分卷后每份压缩文件大小至少64k(65536byte)
如果在对文件(文件夹)分卷压缩前,其已经压缩了,那么将会分卷报错!
那么如何解压呢?对任意一个分卷压缩文件解压即可全部解压;

3.预览压缩文件信息
4.删除功能在下一篇展示,这个在zip文件删除指定文件夹有一种非常巧妙的方法(非常规思路),不需要递归;

代码如下:

 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);
// 这里可以设置文件夹,会自动在压缩文件中创建这些文件夹
// String Folder="1\\2\\";
// para.setFileNameInZip(Folder+fileName);
para.setFileNameInZip(fileName);
// 资源扩展
para.setSourceExternalStream(true);
// 添加文件
zip.addStream(fis, para);
fis.close();
System.out.println("添加成功!");
}
// 分卷,将几个文件进行分卷压缩,一个单位64k
public static void zipTOSplit(String destZipFile, int unit)
throws ZipException {
ZipFile zip = new ZipFile(destZipFile);
ArrayList 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//cbzk//压缩解压//解压之后.txt"));
ZipParameters para = new ZipParameters();
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_FAST);
// 创建分卷
zip.createZipFile(list, para, true, unit * 65536);
System.out.println("分卷成功!");
}
 // 将文件夹进行分卷压缩
public static void zipTOSplit(String destZipFile, String srcFolder, int unit)
throws ZipException {
ZipFile zip = new ZipFile(destZipFile);
ZipParameters para = new ZipParameters();
// 默认COMP_DEFLATE
para.setCompressionMethod(Zip4jConstants.COMP_DEFLATE);
para.setCompressionLevel(Zip4jConstants.DEFLATE_LEVEL_NORMAL);
// 分卷压缩文件夹到指定压缩位置
zip.createZipFileFromFolder(srcFolder, para, true, unit * 65536);
//粗略的算一下分成多少份,获取的大小比实际的大点(一般是准确的)
int partsize=(int)zipInfo(destZipFile)/(unit*64); //65536byte=64kb
System.out.println("分割成功!总共分割成了"+(partsize+1)+"个文件!");
}

// 预览压缩文件信息
public static double zipInfo(String zipFile) throws ZipException {
ZipFile zip = new ZipFile(zipFile);
zip.setFileNameCharset("GBK");
List list = zip.getFileHeaders();
long zipCompressedSize=0;
for(FileHeader head:list){
zipCompressedSize+=head.getCompressedSize();
// System.out.println(zipFile+"文件相关信息如下:");
// 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;//转换为kb
return size;
}

有错误的地方还请更正,谢谢!


推荐阅读
  • Java中包装类的设计原因以及操作方法
    本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • Nginx使用(server参数配置)
    本文介绍了Nginx的使用,重点讲解了server参数配置,包括端口号、主机名、根目录等内容。同时,还介绍了Nginx的反向代理功能。 ... [详细]
  • 搭建Windows Server 2012 R2 IIS8.5+PHP(FastCGI)+MySQL环境的详细步骤
    本文详细介绍了搭建Windows Server 2012 R2 IIS8.5+PHP(FastCGI)+MySQL环境的步骤,包括环境说明、相关软件下载的地址以及所需的插件下载地址。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
    本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • Python正则表达式学习记录及常用方法
    本文记录了学习Python正则表达式的过程,介绍了re模块的常用方法re.search,并解释了rawstring的作用。正则表达式是一种方便检查字符串匹配模式的工具,通过本文的学习可以掌握Python中使用正则表达式的基本方法。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了南邮ctf-web的writeup,包括签到题和md5 collision。在CTF比赛和渗透测试中,可以通过查看源代码、代码注释、页面隐藏元素、超链接和HTTP响应头部来寻找flag或提示信息。利用PHP弱类型,可以发现md5('QNKCDZO')='0e830400451993494058024219903391'和md5('240610708')='0e462097431906509019562988736854'。 ... [详细]
author-avatar
我的生活我做主哦耶_266
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有