我正在使用 ProcessBuilder 来运行 Windows 可执行文件...我需要运行的确切命令是:
"C:\Program Files\CCBU\CCBU.exe" -d"C:\My Data\projects\ccbu\ciccb-report.xls" -tf"C:\Program Files\CCBU\loss-billing-filters.txt"
如果我从命令提示符运行上面的命令,它工作正常。
如果我随后按照以下 StackOverflow 帖子 (ProcessBuilder adds extra quotes to command line) 中的指示将命令和参数作为字符串 [] 数组发出,它会失败,因为目录路径中的空格会以某种方式破坏 CCBU.exe 可执行文件的参数:
[log-snippet]
2015-08-31 10:39:08,937 [main] INFO rpd.primary - C:\Program Files\CCBU\CCBU.exe
logging to the given report's directory
Configuration file is: ./CCBUConfigFile.txt
Running with the following settings:
Report Filepath: C:\My
Search Terms FilePath: C:\Program
2015-08-31 10:39:08,948 [main] INFO rpd.primary - STDERR:--------------------
2015-08-31 10:39:08,961 [main] INFO rpd.primary -
Warning: parameter Data\projects\ccbu\ciccb-report.xls not recognized. Ignoring
Warning: parameter Files\CCBU\loss-billing-filters.txt not recognized. Ignoring
Error: C:\Program not found or not readable
[/log-snippet]
如果我将数据文件和过滤器移动到没有空格的目录路径中,则效果很好:
"C:\Program Files\CCBU\CCBU.exe" -d"C:\Users\n0002501\ccbu\ciccb-report.xls" -tf"C:\Users\n0002501\ccbu\loss-billing-filters.txt"
问题是,此过程的用户会将文件放在确实有空格的文件夹(目录)中。所以我必须以某种方式让它与空间一起工作。我认为这很简单,但我错过了什么?
我正在使用这篇文章中的类来处理 STDOUT 和 STDERR 的线程:http://alvinalexander.com/java/java-exec-processbuilder-process-2
代码如下:
// Split the Arguments :
// In Eclipse and runtime, the arguments get broken :
// The STDOUT from the command shows the Report Filepath
// and Search Teams FilePath as broken at the 1st space...
//
// Report Filepath: C:\My
// Search Terms FilePath: C:\Program
//
// SHOULD BE :
//
// Report Filepath: C:\My Data\projects\ccbu\ciccb-report.xls
// Search Terms FilePath: C:\Program Files\CCBU\loss-billing-filters.txt
//
try {
commands.add ( "\"C:\\Program Files\\CCBU\\CCBU.exe\"" );
commands.add ( "-d\"C:\\My Data\\projects\\ccbu\\ciccb-report.xls\"" );
commands.add ( "-tf\"C:\\Program Files\\CCBU\\loss-billing-filters.txt\"" );
commandExecutor = new SystemCommandExecutor(commands);
commandExecutor.setLog ( getLog() );
// DEBUG : Build and printout the commands...
//
lstrCommand = "";
for ( int theIdx=0; theIdx<commands.size (); theIdx++ ) {
if ( theIdx == 0 ) {
lstrCommand = lstrCommand + commands.get ( theIdx );
}
else {
lstrCommand = lstrCommand + " " + commands.get ( theIdx );
}
getLog().debug ( SHORT_NAME + " Building Command[] [" + commands.get ( theIdx ) + "]" );
}
getLog().debug ( SHORT_NAME + " Running Command[] [" + lstrCommand + "]" );
result = commandExecutor.executeCommand();
// get the stdout and stderr from the command that was run
stdout = commandExecutor.getStandardOutputFromCommand();
stderr = commandExecutor.getStandardErrorFromCommand();
// print the stdout and stderr
getLog().info ( "SystemCommandExecutor - Status Code [" + result + "]" );
getLog().info ( "STDOUT:--------------------" );
getLog().info( stdout );
getLog().info ( "STDERR:--------------------" );
getLog().info( stderr );
}
catch ( Exception ltheXcp ) {
getLog().error ( SHORT_NAME + ".runTask () - Error/exception on commands [3-spaces] [" + lstrCommand + "]" );
}
finally {
commands.clear ();
stdout = null;
stderr = null;
commandExecutor = null;
}
Jayan,最终有效的代码:
try {
commands.add ( "C:\\Program Files\\CCBU\\CCBU.exe" );
commands.add ( "-dC:\\My Data\\projects\\ccbu\\ciccb-report.xls" );
commands.add ( "-tfC:\\Program Files\\CCBU\\loss-billing-filters.txt" );
commandExecutor = new SystemCommandExecutor ( commands );
commandExecutor.setLog ( getLog() );
我所要做的就是去掉所有双引号,让 ProcessBuilder 自己处理目录路径...
阿达姆
最佳答案
添加不带“双”引号的单个字符串..
commands.add ( "C:\\Program Files\\CCBU\\CCBU.exe" );
commands.add ( "-d");
commands.add ("C:\\My Data\\projects\\ccbu\\ciccb-report.xls" );
commands.add ( "-tf");
commands.add("C:\\Program Files\\CCBU\\loss-billing-filters.txt" );
commandExecutor = new SystemCommandExecutor(commands);
ProcessBuilder 将负责必要的参数处理。
拉起评论:
Jayan, You're idea got me thinking : The following worked :
commands.add ( "-dC:\\My Data\\projects\\ccbu\\ciccb-report.xls" ); commands.add ( "-tfC:\\Program Files\\CCBU\\loss-billing-filters.txt"); – lincolnadym
关于Java - 带有空格和双引号的 ProcessBuilder 命令参数失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32314645/
我想用ruby编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序
我正在使用ruby1.9解析以下带有MacRoman字符的csv文件#encoding:ISO-8859-1#csv_parse.csvName,main-dialogue"Marceu","Giveittohimóhe,hiswife."我做了以下解析。require'csv'input_string=File.read("../csv_parse.rb").force_encoding("ISO-8859-1").encode("UTF-8")#=>"Name,main-dialogue\r\n\"Marceu\",\"Giveittohim\x97he,hiswife.\"\
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere
我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"
我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option
我已经构建了一些serverspec代码来在多个主机上运行一组测试。问题是当任何测试失败时,测试会在当前主机停止。即使测试失败,我也希望它继续在所有主机上运行。Rakefile:namespace:specdotask:all=>hosts.map{|h|'spec:'+h.split('.')[0]}hosts.eachdo|host|begindesc"Runserverspecto#{host}"RSpec::Core::RakeTask.new(host)do|t|ENV['TARGET_HOST']=hostt.pattern="spec/cfengine3/*_spec.r
我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano