使用子流程运行命令/程序



我是python中subprocess模块的新手。

文档提供了以下示例:

>>> subprocess.check_output(["echo", "Hello World!"])
b'Hello World!n'

我尝试的是:

>>> import subprocess
>>> subprocess.check_output(["cd", "../tests", "ls"])
/usr/bin/cd: line 4: cd: ../tests: No such file or directory
Traceback (most recent call last):
  File "<input>", line 1, in <module>
  File "/Library/Frameworks/Python.framework/Versions/3.4/lib/python3.4/subprocess.py", line 620, in check_output
    raise CalledProcessError(retcode, process.args, output=output)
subprocess.CalledProcessError: Command '['cd', '../tests', 'ls']' returned non-zero exit status 1

我很困惑,因为这是我的文件结构:

   /proj
      /cron
         test_scheduler.py
      /tests
         printy.py
         test1.py
         test2.py
         ...

这些也是我的其他尝试:

>>> subprocess.check_output(["cd", "../tests", "python", "printy.py"])
/usr/bin/cd: line 4: cd: ../tests: No such file or directory
Traceback (most recent call last):
  File "<input>", line 1, in <module>
  File "/Library/Frameworks/Python.framework/Versions/3.4/lib/python3.4/subprocess.py", line 620, in check_output
    raise CalledProcessError(retcode, process.args, output=output)
subprocess.CalledProcessError: Command '['cd', '../tests', 'python', 'printy.py']' returned non-zero exit status 1

>>> subprocess.check_output(["cd", "../tests;", "ls"])
/usr/bin/cd: line 4: cd: ../tests;: No such file or directory
Traceback (most recent call last):
  File "<input>", line 1, in <module>
  File "/Library/Frameworks/Python.framework/Versions/3.4/lib/python3.4/subprocess.py", line 620, in check_output
    raise CalledProcessError(retcode, process.args, output=output)
subprocess.CalledProcessError: Command '['cd', '../tests;', 'ls']' returned non-zero exit status 1

尽可能避免shell=True

在这种情况下,你当然可以避免。您面临的问题是:cd是一个内置的shell。它不是一个可以从外部调用的命令/程序/实用程序。cd需要在shell中才能工作。相反,您可以更改当前目录。执行命令。然后返回到原始目录。

你需要做如下操作:

pathBefore = os.getcwd()
os.chdir("/path/to/your/directory")
subprocess.check_output(["ls"])
os.chdir(pathBefore) # get back to the path we were in before

UPDATE:@JFSebastian指出的一种更好的方法是对check_output调用使用额外的cwd参数。

tests目录的相对路径取决于脚本的运行位置。我建议你打电话给subprocess.check_output(["pwd"])看看你在哪里。

此外,您不能像尝试使用["cd", "../tests", "python", "printy.py"]那样在同一个调用中组合两个命令。您需要分别使用["cd", "../tests"]["python", "printy.py"]进行两次单独的呼叫。

错误消息很清楚:

/usr/bin/cd:行4:cd:/测试:没有这样的文件或目录

即您已成功启动失败的/usr/bin/cd程序并打印错误消息。

如果您想从../tests目录运行ls命令,请改为:

import os
import subprocess
cwd = os.path.join(get_script_dir(), '../tests')
output = subprocess.check_output(['ls'], cwd=cwd)

其中CCD_ 15。

注意:不要为目录使用相对路径——您的脚本可以从不同的目录运行——在这种情况下,相对路径会失败。

我认为您缺少了一个参数
这里是我写过的唯一一个python脚本的片段:

#!/usr/local/bin/python
from subprocess import call
...
call( "rm " + backupFolder + "*.bz2", shell=True )

请注意通话末尾的shell=True

最新更新