背景:window 下通过压缩软件对文件夹进行压缩后,上传到linux 服务器上,进行解压。
问题:出现中文名名称的文件乱码
尝试解决方案:
1、解压根据 指定编码解压。 #CP936 其实就是GBK(window下的编码,cmd 命令:chcp ,可以看到)。
通过unzip行命令解压,指定字符集。
在阿里云ecs 上按照此方法,可行。但是在私有服务器上,可能由于环境配置问题,未果!!!
unzip -O CP936 xxxx.zip
2.尝试切换 解压方式,例如 tar.gz、7z 等方式后,依然不好使
3.在window下针对 压缩文件源 使用 java 程序,指定编码 进行 压缩。然后将压缩包上传至 linux ,直接解压,中文名可以正常显示。
代码如下:
package com.rexel;
import org.apache.tools.zip.ZipEntry;
import org.apache.tools.zip.ZipOutputStream;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.*;
/**
* @ClassName CompressTest
* @Description TODO
* @Author Hai.Dong
* @Date 2021/5/12 14:22
**/
public class CompressTest {
private final static Logger logger = LoggerFactory.getLogger(CompressTest.class);
private static final int BUFFER_SIZE = 2 * 1024;
public static void main(String[] args) throws Exception {
//调用 toZip 方法,进行 指定 编码 压缩
}
/**
* 压缩成ZIP
*
* @param srcFilePath 压缩文件路径
* @param tarFilePath 目标ZIP输出路径
* @param KeepDirStructure 是否保留原来的目录结构,true:保留目录结构;
* false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)
* @throws Exception 压缩失败会抛出异常
*/
public static boolean toZip(String srcFilePath, String tarFilePath, boolean KeepDirStructure) throws Exception {
boolean isCompressSuccess = false;
long start = System.currentTimeMillis();
FileOutputStream fos = null;
ZipOutputStream zos = null;
try {
File sourceFile = new File(srcFilePath);
if (!sourceFile.exists()) {
throw new FileNotFoundException("待压缩文件 [" + srcFilePath + "]不存在.");
}
fos = new FileOutputStream(new File(tarFilePath));
zos = new ZipOutputStream(fos);
// 设置压缩的编码,解决压缩路径中的中文乱码问题
zos.setEncoding("UTF-8");
compress(sourceFile, zos, sourceFile.getName(), KeepDirStructure);
isCompressSuccess = true;
long end = System.currentTimeMillis();
logger.info("【文件压缩】 压缩完成,耗时:{} ms", (end - start));
} catch (Exception e) {
logger.error("【文件压缩】 压缩失败", e);
throw new RuntimeException("文件压缩失败", e);
} finally {
closeOutPutStream(zos);
closeOutPutStream(fos);
}
return isCompressSuccess;
}
/**
* 递归压缩方法
*
* @param sourceFile 源文件
* @param zos zip输出流
* @param name 压缩后的名称
* @param KeepDirStructure 是否保留原来的目录结构,true:保留目录结构;
* false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)
* @throws Exception
*/
private static void compress(File sourceFile, ZipOutputStream zos, String name, boolean KeepDirStructure)
throws Exception {
byte[] buf = new byte[BUFFER_SIZE];
if (sourceFile.isFile()) {
// 向zip输出流中添加一个zip实体,构造器中name为zip实体的文件的名字
zos.putNextEntry(new ZipEntry(name));
// copy文件到zip输出流中
int len;
FileInputStream in = new FileInputStream(sourceFile);
while ((len = in.read(buf)) != -1) {
zos.write(buf, 0, len);
}
zos.closeEntry();
in.close();
} else {
File[] listFiles = sourceFile.listFiles();
if (listFiles == null || listFiles.length == 0) {
// 需要保留原来的文件结构时,需要对空文件夹进行处理
if (KeepDirStructure) {
// 空文件夹的处理
zos.putNextEntry(new ZipEntry(name + "/"));
// 没有文件,不需要文件的copy
zos.closeEntry();
}
} else {
for (File file : listFiles) {
// 判断是否需要保留原来的文件结构
if (KeepDirStructure) {
// 注意:file.getName()前面需要带上父文件夹的名字加一斜杠,
// 不然最后压缩包中就不能保留原来的文件结构,即:所有文件都跑到压缩包根目录下了
compress(file, zos, name + "/" + file.getName(), KeepDirStructure);
} else {
compress(file, zos, file.getName(), KeepDirStructure);
}
}
}
}
}
/**
* 释放资源
*
* @param ops
* @return void
* @Title closeOutPutStream
*/
public static void closeOutPutStream(OutputStream ops) {
if (ops != null) {
try {
ops.close();
} catch (IOException ex) {
logger.error("关闭输出流失败", ex);
}
}
}
}