Python OSError: [Errno 2]

38
我有以下代码,试图在Linux中启动每个“命令”。该模块尝试保持这两个命令运行,如果其中任何一个因任何原因崩溃。
#!/usr/bin/env python
import subprocess

commands = [ ["screen -dmS RealmD top"], ["screen -DmS RealmD top -d 5"] ]
programs = [ subprocess.Popen(c) for c in commands ]
while True:
    for i in range(len(programs)):
        if programs[i].returncode is None:
            continue # still running
        else:
            # restart this one
            programs[i]= subprocess.Popen(commands[i])
        time.sleep(1.0)

执行代码时,抛出以下异常:
Traceback (most recent call last):
  File "./marp.py", line 82, in <module>
    programs = [ subprocess.Popen(c) for c in commands ]
  File "/usr/lib/python2.6/subprocess.py", line 595, in __init__
    errread, errwrite)
  File "/usr/lib/python2.6/subprocess.py", line 1092, in _execute_child
    raise child_exception
OSError: [Errno 2] No such file or directory

我觉得我漏掉了一些显而易见的东西,有人能看出上面的代码有什么问题吗?


2
你应该使用 programs[i].poll() 而不是 programs[i].returncode - jfs
6个回答

66

使用 ["screen", "-dmS", "RealmD", "top"] 代替 ["screen -dmS RealmD top"]

或许还可以使用完整的路径来调用 screen 命令。


1
文档中说可以使用字符串或序列。 - Colin Burnett
命令 = [["screen", "-dmS", "RealmD", "top"], ["screen", "-DmS", "RealmD", "top", "-d", "5"]]完美地工作了! - Caedis
@colin:在这种情况下,使用了一个序列,并且它需要单独包含参数。可能一个简单的字符串(不带[])也可以工作。 - sth
1
每个列表元素可以是一个字符串或一个参数“word”。如果您传递一个序列,您应该已经完成了shell通常执行的精确解析。如果您传递单个字符串,则它将被解析为shell将要解析的方式。 - S.Lott
@umpirsky:从文档中可以看到:“在Unix上,如果args是一个字符串,则该字符串被解释为要执行的程序的名称或路径。但是,只有在不传递参数给程序时才能这样做。” - jfs
我该如何使用 & 符号?就像这样:cmd_list=['dd','if=/dev/zero','of=/dev/null','&'] - mihir6692

9

唯一的猜测是找不到 screen。尝试使用 /usr/bin/screen 或者 which screen 返回的任何路径。


是的,这就是我的问题。命令没有按预期添加到系统路径中。 - Brian Moeskau

7
问题在于你的命令应该被分割。subprocess要求cmd是一个列表,而不是一个字符串。 它不应该是这样的:
subprocess.call('''awk 'BEGIN {FS="\t";OFS="\n"} {a[$1]=a [$1] OFS $2 FS $3 FS $4} END
{for (i in a) {print i a[i]}}' 2_lcsorted.txt > 2_locus_2.txt''') 

那样是行不通的。如果你向subprocess传递一个字符串,它会认为这是要执行的命令的路径。命令需要是一个列表。请查看http://www.gossamer-threads.com/lists/python/python/724330。此外,因为你正在使用文件重定向,应该使用subprocess.call(cmd, shell=True)。你也可以使用shlex


假设它在shell的路径中,subprocess.call("executable -parameter 1 -hello", shell=True)是有效的。 - WoJ
这取决于调用方式。如果您不解析输入,则与 shell 命令一样安全。 - WoJ

3
我写了类似这样的代码时,也遇到了相同的错误:
subprocess.Popen("ls" ,shell = False , stdout = subprocess.PIPE ,stderr = subprocess.PIPE)

问题已经解决了,当我设置shell=True时。它可以正常工作。 subprocess.Popen("ls" ,shell = False , stdout = subprocess.PIPE ,stderr = subprocess.PIPE, shell=True)

2
commands = [ "screen -dmS RealmD top", "screen -DmS RealmD top -d 5" ]
programs = [ subprocess.Popen(c.split()) for c in commands ]

6
请使用 shlex.split() 替代 string.split() - Tanky Woo

0

以防万一..我也遇到了这个错误,问题是我的文件是DOS格式而不是UNIX格式,所以在此处出现了问题:

 return subprocess.call(lst_exp)

其中lst_exp是参数列表,其中一个参数因为在DOS而不是UNIX中导致其为“未找到”,但抛出的错误相同:

File "/var/www/run_verifier.py", line 59, in main
return subprocess.call(lst_exp)
File "/usr/lib/python2.7/subprocess.py", line 522, in call
return Popen(*popenargs, **kwargs).wait()
File "/usr/lib/python2.7/subprocess.py", line 710, in __init__
errread, errwrite)
File "/usr/lib/python2.7/subprocess.py", line 1335, in _execute_child
raise child_exception
OSError: [Errno 2] No such file or directory

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接