Python子进程通配符用法

问题描述 投票:52回答:2
import os

import subprocess

proc = subprocess.Popen(['ls','*.bc'], stdout=subprocess.PIPE, stderr=subprocess.PIPE)

out,err = proc.communicate()

print out

此脚本应打印带有.bc后缀的所有文件,但它会返回一个空列表。如果我在命令行中手动执行ls * .bc它可以工作。在脚本中执行['ls','test.bc']也可以,但由于某些原因,星形符号不起作用..任何想法?

python subprocess wildcard
2个回答
61
投票

您需要提供shell=True以通过shell解释器执行命令。但是,如果这样做,则不能再提供列表作为第一个参数,因为参数将被引用。而是指定原始命令行,因为您希望将其传递给shell:

 proc = subprocess.Popen('ls *.bc', shell=True,
                                    stdout=subprocess.PIPE,
                                    stderr=subprocess.PIPE)

50
投票

扩展* glob是shell的一部分,但默认情况下subprocess不会通过shell发送命令,因此执行命令(第一个参数ls),然后使用文字*作为参数。

这是一件好事,请参阅子流程文档的the warning block in the "Frequently Used Arguments" section。它主要讨论安全问题,但也可以帮助避免愚蠢的编程错误(因为没有魔术shell字符需要担心)

我对shell=True的主要抱怨是它通常意味着有一个更好的方法来解决这个问题 - 用你的例子,你应该使用the glob module

import glob
files = glob.glob("*.bc")
print files # ['file1.bc', 'file2.bc']

这将更快(没有进程启动开销),更可靠和跨平台(不依赖于具有ls命令的平台)

© www.soinside.com 2019 - 2024. All rights reserved.