我正在尝试编写一个python脚本,将一个字符串发送到程序并将其放在后台。在命令行中,我可以复制并粘贴以下代码,并使其成功执行我想要的python程序:
printf "f\nil\ncs\n1.e-8 100.0 1.e-8\nn\n0.002\nb\n0.05\nz\n2.e-7\nx4\n5.e-7\n\n\nfort.13\nn\nn\n\n" | vpfit95
其中:vpfit95
是PATH中可执行程序的别名。
我尝试过的一些排列(一次尝试一次):
import subprocess
vpfitExecutable = 'vpfit95'
String1=r'f\nil\ncs\n1.e-8 100.0 1.e-8\nn\n0.002\nb\n0.05\nz\n2.e-7\nx4\n5.e-7\n\n\nfort.13\nn\nn\n\n'
String2=r"f\nil\ncs\n1.e-8 100.0 1.e-8\nn\n0.002\nb\n0.05\nz\n2.e-7\nx4\n5.e-7\n\n\nfort.13\nn\nn\n\n"
String3="f\nil\ncs\n1.e-8 100.0 1.e-8\nn\n0.002\nb\n0.05\nz\n2.e-7\nx4\n5.e-7\n\n\nfort.13\nn\nn\n\n"
cmd1 = "printf \"" + String1 + "\""
cmd2 = "printf \"" + String1 + "\" | " + vpfitExecutable
cmd3 = "printf \"" + String2 + "\""
cmd4 = "printf \"" + String3 + "\""
print cmd2
p1 = subprocess.Popen([vpfitExecutable, cmd1])
p2 = subprocess.Popen([cmd2])
p3 = subprocess.Popen([vpfitExecutable, cmd3])
p4 = subprocess.Popen([vpfitExecutable, cmd4])
p4 = subprocess.Popen([vpfitExecutable, String1])
p5 = subprocess.Popen([vpfitExecutable, String2])
p6 = subprocess.Popen([vpfitExecutable, String3])
# check
p7 = subprocess.Popen([vpfitExecutable]) # works.
所有失败(除了p7)。有些失败了“Fortran运行时错误:文件结束”(来自vpfit程序)。其他人失败了一个Traceback到子进程库和“OSError:[Errno 2]没有这样的文件或目录”。
一些检查:当我打印cmd2
并将其复制并粘贴到终端时,它可以完美地工作。当我运行p7
时,它正确地开始按预期运行程序,只是没有输入字符串。我显然在这里缺少一些基本的东西,而我却无法弄清楚是什么。任何帮助表示赞赏!
答案 0 :(得分:6)
是的,你缺少一些基本的东西 - 管道由shell处理。要使用子流程执行所需操作,应将vpfit95
作为子流程运行,然后使用communicate
运行。请参阅文档的第17.1.4.2节replacing a shell pipeline。
proc = subprocess.Popen([vpfitExecutable], stdin=subprocess.PIPE)
proc.communicate(input='your string')
如果需要检查vpfit的stdout和stderr,则从communicate
的输出元组中获取句柄。
一个脏的替代方法是使用shell=True
运行命令,它在shell中生成它然后你可以在命令中使用管道,但是“切断中间人”并与之交互更清晰直接子进程。