ZipInputStream解压报错java.lang.IllegalArgumentException: MALFORMED

背景

使用jdk自带的zip工具ZipInputStream,去读取使用winrar在中文windows制作出来的zip文件报错。

c 复制代码
Exception in thread "main" java.lang.IllegalArgumentException: MALFORMED
	at java.util.zip.ZipCoder.toString(ZipCoder.java:58)
	at java.util.zip.ZipInputStream.readLOC(ZipInputStream.java:300)
	at java.util.zip.ZipInputStream.getNextEntry(ZipInputStream.java:122)

从报错信息"MALFORMED",第一反应就是文件名解码失败,因为使用winrar制作的zip肯定是符合zip标准格式的,唯一有疑点的就是文件名是中文,然后存储时的编码和解码是的编码不一致。

查看ZipInputStream构造函数可以传编码,因此使用

new ZipInputStream(new FileInputStream(zipFile),Charset.forName("GBK"))尝试解码,结果成功,因此确认报错由文件名解码失败导致。

改进

本系统服务全球,并不是所有人都是用的是中文操作系统,ZIP文件的编码格式并不是唯一的,因此需要找到更智能zip工具替代ZipInputStream。

以前做压缩算法时,有听说过Apache的commons compress,因此尝试一下,该工具能够自动识别编码,做到对文件名的正确解压。

尝试使用ZipArchiveInputStream,在不传编码格式的情况下,能够正常解码出文件名。好奇它是如何做到的,因此翻阅了相关,文档及源码结论如下:

  1. 使用传入的编码尝试解码文件名,解码器中对于无法识别的字节使用"?"替代(对比JDK字节报错,该设计,作为开发应该更赞同这种方式吧)。
  2. 尝试对额外的文件名字段,使用UTF-8解码,该值优先级更高,只要能解码出非空的值,就优先使用该文件名。

    上述两处代码跟着ZipArchiveInputStream的getNextZipEntry()方法就可以跟踪到。

关于上述第2点的说明:

可以阅读https://commons.apache.org/proper/commons-compress/zip.html中的Encoding章节。

主要内容如下,

1、最开始zip使用CodePage 437作为文件名的编码,这个字符集太小无法满足全球所有国家的使用

2、然后不同的打包工具开始尝试使用不同的方式来支持不同文字的文件名。有一种方式就是使用传入的编码对文件名进行编码并存入到zip规定的文件名字段中,并使用zip定义的扩展字段存储,使用UTF-8编码的文件名。当然并不是所有的工具都会这么干,所以commons compress也支持尝试使用该字段解码文件名,如果为空,还是会使用文件名的标准字段解码出的文件名。

相关推荐
shinelord明4 分钟前
【再谈设计模式】享元模式~对象共享的优化妙手
开发语言·数据结构·算法·设计模式·软件工程
新手小袁_J9 分钟前
JDK11下载安装和配置超详细过程
java·spring cloud·jdk·maven·mybatis·jdk11
呆呆小雅10 分钟前
C#关键字volatile
java·redis·c#
Monly2111 分钟前
Java(若依):修改Tomcat的版本
java·开发语言·tomcat
boligongzhu12 分钟前
DALSA工业相机SDK二次开发(图像采集及保存)C#版
开发语言·c#·dalsa
Eric.Lee202112 分钟前
moviepy将图片序列制作成视频并加载字幕 - python 实现
开发语言·python·音视频·moviepy·字幕视频合成·图像制作为视频
Ttang2313 分钟前
Tomcat原理(6)——tomcat完整实现
java·tomcat
7yewh14 分钟前
嵌入式Linux QT+OpenCV基于人脸识别的考勤系统 项目
linux·开发语言·arm开发·驱动开发·qt·opencv·嵌入式linux
钱多多_qdd24 分钟前
spring cache源码解析(四)——从@EnableCaching开始来阅读源码
java·spring boot·spring
waicsdn_haha26 分钟前
Java/JDK下载、安装及环境配置超详细教程【Windows10、macOS和Linux图文详解】
java·运维·服务器·开发语言·windows·后端·jdk