使用java中的ProcessBuilder提取7zip cmd



我在使用Java 10 ProcessBuilder和带有命令行的7z.exe(18.05(将存档提取到所需类别时遇到问题。当我使用Windows CMD时,完全相同的命令可以正常工作,但当我的JavaFX应用程序使用ProcessBuilder:发出时,该命令不再起作用

public static void decompress7ZipEmbedded(File source, File destination) throws IOException, InterruptedException {
ProcessBuilder pb = new ProcessBuilder(
getSevenZipExecutablePath(),
EXTRACT_WITH_FULL_PATHS_COMMAND,
quotifyPath(source.getAbsolutePath()),
OUTPUT_DIRECTORY_SWITCH + quotifyPath(destination.getAbsolutePath())
);
processWithSevenZipEmbedded(pb);
}
private static void processWithSevenZipEmbedded(ProcessBuilder pb) throws IOException, InterruptedException {
LOG.info("7-zip command issued: " + String.join(" ", pb.command()));
Process p = pb.start();
new Thread(new InputConsumer(p.getInputStream())).start();
System.out.println("Exited with: " + p.waitFor());
}
public static class InputConsumer implements Runnable {
private InputStream is;
InputConsumer(InputStream is) {
this.is = is;
}
@Override
public void run() {
try {
int value = -1;
while ((value = is.read()) != -1) {
System.out.print((char) value);
}
} catch (IOException exp) {
exp.printStackTrace();
}
LOG.debug("Output stream completed");
}
}
public static String getSevenZipExecutablePath() {
return FileUtil.quotifyPath(getDirectory() + "7z" + "/" + "7z");
}
public static String quotifyPath(String path) {
return '"' + path + '"';
}
public class Commands {
public static final String EXTRACT_COMMAND = "e";
public static final String EXTRACT_WITH_FULL_PATHS_COMMAND = "x";
public static final String PACK_COMMAND = "a";
public static final String DELETE_COMMAND = "d";
public static final String BENCHMARK_COMMAND = "b";
public static final String LIST_COMMAND = "l";
}
public class Switches {
public static final String OUTPUT_DIRECTORY_SWITCH = "-o";
public static final String RECURSIVE_SWITCH = "-r";
public static final String ASSUME_YES = "y";
}

命令如下:

"C:/Users/blood/java_projects/AppRack/target/classes/7z/7z" x "D:PulpitAppRack SandboxtestsomethingSomething 2Something2.7z" -o"D:PulpitAppRack SandboxSomething2"

以及ProcessBuilder的输出:

7-Zip 18.05 (x64) : Copyright (c) 1999-2018 Igor Pavlov : 2018-04-30
Scanning the drive for archives:
1 file, 59177077 bytes (57 MiB)
Extracting archive: D:PulpitAppRack SandboxtestSomethingSomething 2Something2.7z
--
Path = D:PulpitAppRack SandboxtestSomethingSomething 2Something2.7z
Type = 7z
Physical Size = 5917Exited with: 0
7077
Headers Size = 373
Method = LZMA2:26 LZMA:20 BCJ2
Solid = +
Blocks = 2

No files to process
Everything is Ok
Files: 0
Size:       0
Compressed: 59177077

它没有任何作用。没有创建所需的文件夹,什么都没有。使用CMD,它就像一个魅力(这里使用相同的命令从Windows 10 CMD日志(:

7-Zip 18.05 (x64) : Copyright (c) 1999-2018 Igor Pavlov : 2018-04-30
Scanning the drive for archives:
1 file, 59177077 bytes (57 MiB)
Extracting archive: D:PulpitAppRack SandboxtestSomethingSomething 2Something2.7z
--
Path = D:PulpitAppRack SandboxtestSomethingSomething 2Something2.7z
Type = 7z
Physical Size = 59177077
Headers Size = 373
Method = LZMA2:26 LZMA:20 BCJ2
Solid = +
Blocks = 2
Everything is Ok
Folders: 1
Files: 5
Size:       64838062
Compressed: 59177077

你知道是什么导致了这里的差异吗?为什么它什么都不做就说"没有文件要处理,一切都好"?我已经尝试过首先使用File类创建文件夹,但这似乎不是问题,因为无论目标文件夹在提取之前是否存在,结果都是一样的。

我已经尝试了我脑海中的一切,但我现在没有什么想法了。请与我分享你对这个问题的任何建议。非常感谢。

非常感谢您的帮助。

不要引用你的论点。引号是为了命令shell的利益。ProcessBuilder不是命令shell;它直接执行命令,因此任何引号都被视为参数本身(即文件名(的一部分。此外,pb.inheritIO((;是查看子进程输出的更好方式,而不是手动消耗进程流。

谢谢@VGR,这似乎是个问题——在我删除了上述命令中引用路径的方法后,它就像一个魅力,可以毫无问题地提取存档!因此得出的结论是,在使用Java ProcessBuilder时,我不应该在路径中使用引号。

我也使用过pb.inheritIO((,你是对的,用这种方式管理它会更好、更容易。

public static void decompress7ZipEmbedded(File source, File destination) throws IOException {
ProcessBuilder pb = new ProcessBuilder().inheritIO().command(
getSevenZipExecutablePath(),
EXTRACT_WITH_FULL_PATHS_COMMAND,
source.getAbsolutePath(),
OUTPUT_DIRECTORY_SWITCH + destination.getAbsolutePath(),
OVERWRITE_WITHOUT_PROMPT
);
processWithSevenZipEmbedded(pb);
}
private static void processWithSevenZipEmbedded(ProcessBuilder pb) throws IOException {
LOG.info("7-zip command issued: " + String.join(" ", pb.command()));
pb.start();
}
public class Commands {
public static final String EXTRACT_WITH_FULL_PATHS_COMMAND = "x"; 
}
public class Switches {
public static final String OUTPUT_DIRECTORY_SWITCH = "-o";
public static final String OVERWRITE_WITHOUT_PROMPT = "-aoa";
}

双击文件7zip.chm或启动7-Zip并打开帮助并阅读帮助页命令行版本-第一行7z[…][…]的语法。有明确的解释,首先必须指定命令x,其次应该是开关,如-o,最好的最后一个开关是--,然后是存档文件名和最后一个进一步的参数,如要提取的文件/文件夹的名称。开关也可以在归档文件名之后指定,但不建议这样做,尽管帮助页面上的-o示例也在末尾使用-o。

感谢@Mofi的提示。我使用了-aoa开关而不是-y,它终于开始按我的意愿工作了——在没有任何提示的情况下覆盖文件。我让命令的其余部分保持原样,因为它按预期工作,所以它最终看起来是这样的:

C:/Users/blood/java_projects/AppRack/target/classes/7z/7z" x D:PulpitAppRack SandboxtestTestTest 2Test.7z -oD:DesktopAppRack SandboxTest 2 -aoa

再次感谢您的帮助!

最新更新