0

我正在尝试解压缩一些 zip 文件,它大约有 65 兆。下面的代码片段:

这个方法实际上解压了一个文件:

public synchronized void execute(Path zipFile) {
    final ProcessBuilder builder = new ProcessBuilder("/bin/unzip",zipFile.toAbsolutePath().toString(), "-d", dir.toAbsolutePath().toString());
    FutureTask<Integer> task = new FutureTask<Integer>(new Callable<Integer>() {
        @Override public Integer call() {
            try {
                System.out.println("started and waiting");
                int status = builder.start().waitFor();
                System.out.println("status: " + status);
                return status;
            } catch (InterruptedException e) {
            } catch (IOException e) {
            }
            return 0;
        }
    });

    List<FutureTask<Integer>> tasks = new ArrayList<FutureTask<Integer>>();
    tasks.add(task);
    System.out.println("task added");
    ExecutorService executor = Executors.newCachedThreadPool();
    for (FutureTask<Integer> t : tasks) {
        executor.submit(t);
        System.out.println("submitted");
    }
    executor.shutdown();
    System.out.println("shutdown");
}

那个执行者/未来的东西只是为了确保我做得正确。这个方法在类 Finder 中被调用,它在目录中找到 zip 文件并尝试解压它。它基于此代码http://docs.oracle.com/javase/tutorial/essential/io/walk.html

所以具体来说:

@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) {
    if (find(file)) {
        synchronized(Finder.class) {
            executor.execute(file);
        }
    }
    return CONTINUE;
}

现在的问题。这真的很有趣。每当我通过此代码提取某些内容时,zip 文件实际上会被解压缩,但只有一些目录被解压缩,而其他目录则没有。例如,我有一个包含 temp foo 和 bar 目录的 zip 文件,但解压缩后,只有 temp 和 foo 目录,并且 bar 没有被提取。

输出是:

task added
submitted
started and waiting
shutdown

为什么没有“status = something”输出?

我不明白为什么会这样。当我手动解压缩时,它会正确解压缩。

// 编辑

这成功了

@Override
public synchronized void execute(String file, String dest) {
    ProcessBuilder pb = new ProcessBuilder("/bin/unzip","-qq", file, "-d", dest);
    pb.redirectErrorStream(true);

    try {
        Process p = pb.start();
        InputStream is = p.getInputStream();
        InputStreamReader r = new InputStreamReader(is);
        BufferedReader in = new BufferedReader(r);
        String line;
        while ((line = in.readLine()) != null) {
            // because of the -qq option, it does actually write out nothing
            System.out.println(line);
        }
    } catch (IOException ex) {
        System.err.println(ex.getMessage());
    }
}
4

2 回答 2

2

unzip命令将它正在解压缩的文件的详细信息打印到其标准输出,因此您需要在 Java 程序中阅读它(通过Process.getInputStream)。如果您没有及时读取输出,则进程可能会在其缓冲区满时阻塞 - 这在 Process 的 javadoc 中有详细说明。

我建议您打电话builder.redirectErrorStream(true),然后确保您从流程流中读取所有数据。您还可以从向-qqunzip 调用添加参数中受益,以最大限度地减少它首先创建的输出量。

于 2012-09-29T17:34:38.583 回答
0

我认为问题是由于超时限制。

您可以尝试下面的代码来执行过程

Runtime runtime = Runtime.getRuntime();
Process process = runtime.exec(commandLine);

我认为它会起作用。

于 2012-09-29T17:20:02.743 回答