gpt4 book ai didi

java - ProcessBuilder/Runtime.exec() 与 Weka 命令行演示特殊行为

转载 作者:行者123 更新时间:2023-11-30 06:03:44 28 4
gpt4 key购买 nike

下面基本上是我的完整问题的 MCVE,这要困惑得多。您需要知道的是,直接放入终端时会运行以下行:

java -classpath /path/to/weka.jar weka.filters.MultiFilter \
-F "weka.filters.unsupervised.attribute.ClusterMembership -I first" \
-i /path/to/in.arff

这相对简单。基本上,我所做的就是尝试使用 ClusterMembership 过滤器的所有默认设置对 in.arff 中的数据进行聚类,但我想忽略第一个属性。我在那里有 MultiFilter,因为在我的实际项目中,还有其他过滤器,所以我需要保留它。正如前面提到的,这工作得很好。但是,当我尝试使用 ProcessBuilder 运行同一行时,我收到“引用解析错误”,并且嵌套引号的整个结构似乎崩溃了。证明这一点的一种方法是尝试让以下内容发挥作用:

List<String> args = new ArrayList<String>();
args.add("java");
args.add("-cp");
args.add("/path/to/weka.jar");
args.add("weka.filters.MultiFilter");
args.add("-F");
args.add("\"weka.filters.unsupervised.attribute.ClusterMembership");
args.add("-I");
args.add("first\"");
args.add("-i");
args.add("/path/to/in.arff");
ProcessBuilder pb = new ProcessBuiler(args);

// ... Run the process below

乍一看,您可能会认为这与上面的行相同(这当然是我天真的想法)。事实上,如果我只是打印 args 并在每个参数之间留有空格,则生成的字符串是相同的,并且如果直接复制并粘贴到终端,则可以完美运行。但是,无论出于何种原因,当我收到消息(来自 Weka)引用解析错误时,该程序将无法运行。我尝试谷歌搜索并发现this关于 ProcessBuilder 如何向命令行添加额外引号的问题(这导致我尝试了多种转义序列组合,但所有这些组合都不起作用),并阅读 this关于 ProcessBuilder/Runtime.exec() 如何工作的文章(我尝试了 ProcessBuilder 和 Runtime.exec(),最终同样的问题仍然存在),但找不到与我需要的相关的任何内容。 Weka 已经有了糟糕的文档,然后他们的 Wikispace 页面在几周前由于 Wikispaces 关闭而关闭了,所以我在 Weka 方面发现的信息非常少。

我的问题是:有没有一种方法可以让我上面提到的第二个示例类似的东西运行,这样我就可以将参数组合在一起以获得更大的命令?我知道它可能需要一些时髦的转义序列(或者可能不需要?),或者可能是我没有考虑过的其他东西。非常感谢这里的任何帮助。

编辑:我更新了问题,希望能更深入地了解我的问题所在。

最佳答案

您不需要将参数组合在一起。正如您已经注意到的,它甚至不起作用。看看当我像这样调用 Java 程序时会发生什么:

java -jar Test.jar -i -s "-t 500"

这是我的“程序”:

public class Test {
public static void main(String[] args) {
for( String arg : args ) {
System.out.println(arg);
}
}
}

这是输出:

-i
-s
-t 500

参数中不包含引号,它们用于对参数进行分组。因此,当您像您一样将参数传递给 ProcessBuilder 时,本质上就像您在命令行上用引号编写它们一样,并且它们被视为单个参数,这会使解析器感到困惑。

仅当您有嵌套组件时才需要引号,例如过滤分类器。也许my answer on another Weka question可以帮助您处理这些嵌套组件。 (我最近将他们的 wiki 的链接更改为指向 Google 缓存,直到他们建立了新的 wiki。)

由于您没有具体说明具体是什么情况导致您考虑分组,因此您可以尝试为 Weka 获取一个工作命令行,然后使用该命令行作为像我这样的程序的输入。然后您可以了解如何将它们传递给 ProcessBuilder

对于您的示例,我猜想以下内容会起作用:

List<String> args = new ArrayList<String>();
args.add("java");
args.add("-cp");
args.add("/path/to/weka.jar");
args.add("weka.filters.MultiFilter");
args.add("-F");
args.add("weka.filters.unsupervised.attribute.ClusterMembership -I first");
args.add("-i");
args.add("/path/to/in.arff");
ProcessBuilder pb = new ProcessBuiler(args);

其他详细信息

Weka 内部发生的事情基本上如下:参数中的选项首先由 weka.filters.Filter 处理,然后所有非通用过滤器选项由 weka.filters 处理.MultiFilter,其中在setOptions(...)中包含以下代码:

filters = new Vector<Filter>();
while ((tmpStr = Utils.getOption("F", options)).length() != 0) {
options2 = Utils.splitOptions(tmpStr);
filter = options2[0];
options2[0] = "";
filters.add((Filter) Utils.forName(Filter.class, filter, options2));
}

这里,tmpStr-F 选项的值,将由 Utils.splitOption(tmpStr) 处理 ( source code )。在那里,所有引用和取消引用的魔法都会发生,以便下一个组件将收到一个选项数组,看起来就像它是第一级组件一样。

关于java - ProcessBuilder/Runtime.exec() 与 Weka 命令行演示特殊行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51768461/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com