1

好吧,我的问题很简单,是关于一个意外的行为(或者至少对我来说是意外的),当我尝试压缩一个目录时,我有以下我自己创建的方法(我很清楚我'没有处理异常和所有那些东西,这是因为(到目前为止)我只是这样做是为了学习如何做到这一点,所以稳定性“并不重要”),这里是代码:

public static void zipDirectory(File srcDirectory, File zipFile) throws IllegalArgumentException {
    if (!srcDirectory.isDirectory()) {
        throw new IllegalArgumentException("The first parameter (srcDirectory) MUST be a directory.");
    }
    int bytesRead;
    byte[] dataRead  = new byte[1000];
    BufferedInputStream in = null;
    ZipOutputStream zOut;
    try {
        zOut = new ZipOutputStream(new BufferedOutputStream(new FileOutputStream(zipFile)));
        for (File f : srcDirectory.listFiles()) {
            if (f.isDirectory()) {
                FileUtilities.zipInnerDirectory(f,zOut);
            }else {
                in = new BufferedInputStream(new FileInputStream(f.getAbsolutePath()), 1000);
                zOut.putNextEntry(new ZipEntry(f.getPath())); 
                while((bytesRead = in.read(dataRead,0,1000)) != -1) {
                    zOut.write(dataRead, 0, bytesRead);
                }
                zOut.closeEntry();
            }
        }
        zOut.flush();
        zOut.close();
    } catch (FileNotFoundException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

private static void zipInnerDirectory(File dir, ZipOutputStream zOut) throws IllegalArgumentException {
    if (!dir.isDirectory()) {
        throw new IllegalArgumentException("The first parameter (srcDirectory) MUST be a directory.");
    }
    BufferedInputStream in = null;
    int bytesRead;
    byte[] dataRead  = new byte[1000];
    try {
        for (File f : dir.listFiles()) {
            if (f.isDirectory()) {
                FileUtilities.zipInnerDirectory(f,zOut);
            }else {
                in = new BufferedInputStream(new FileInputStream(f.getAbsolutePath()), 1000);
                zOut.putNextEntry(new ZipEntry(f.getPath())); 
                while((bytesRead = in.read(dataRead,0,1000)) != -1) {
                    zOut.write(dataRead, 0, bytesRead);
                }
                zOut.closeEntry();
            }
        }
        zOut.flush();
        zOut.close();
    } catch (FileNotFoundException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

正如我所说的不是我最好的编码,所以请不要评判代码(或者至少不要太严格;)),我知道它可以更好;好的“意外行为”是这样的,假设我有以下目录:

  • H:\MyDir1\MyDir2\MyDirToZip

当我将使用该路径创建的文件作为参数发送new File("H:\\MyDir1\\MyDir2\\MyDirToZip")时(

  • H:\MyDir1\MyDir2\MyDirToZip

当我期待在里面找到时:

  • \MyDirToZip

没有 H: \MyDir1 \MyDir2 这是“不必要的”(顺便说一句,它们只是以适当的顺序相互包含一个,我的意思是,其中的其他文件没有被压缩,这就是为什么我说它们是不必要的)所以问题是,我做错了什么?如何指定我只想将结构压缩到srcDirectory

4

1 回答 1

2
zOut.putNextEntry(new ZipEntry(f.getPath())); 

这应该是问题所在。f.getPath()将返回相对于某个根目录(可能是您当前的工作目录)的路径,但不相对于您正在压缩的目录。您需要找出一种从 zip 目录中获取相对路径的方法,可能会这样做:

new ZipEntry(f.getAbsolutePath().substring(zipDir.getAbsolutePath().length()))

或者,如果要添加根目录:

new ZipEntry(zipDir.getName() + "/"
             + f.getAbsolutePath().substring(zipDir.getAbsolutePath().length()))
于 2012-11-16T15:59:31.927 回答