3

我正在尝试使用子进程库和 Popen 方法将文件从一个位置复制到另一个位置。运行以下脚本时出现错误cp: cannot stat /some/dev_path/*。有人告诉我,*它没有扩展到文件名,这就是问题所在。同样在其他一些帖子中,人们建议使用 call 而不是 Popen,但据我所知 call 不会返回 stderr。

devPath = '/some/dev_path/'
productionPath = '/some/prod_path/'

p = subprocess.Popen(['cp', '-r', devPath + '*', productionPath], stdout = subprocess.PIPE, stderr = subprocess.PIPE)
pout, perr = p.communicate()

if perr != '':
    sys.exit('Error: ' + perr)
4

1 回答 1

13

扩展*(globbing) 是你的 shell 的一个功能,例如bash 。因此,您必须shell=Truesubprocess.Popen调用中使用关键字参数。

但是,对于这种情况,我强烈建议改用shutil.copytree

(首先,因为它更简单(参见Zen of Python)并且不易出错。处理错误更加简洁,你会得到很好的异常,包括错误列表(对于像你这样的多文件操作),而你不需要不必处理生成子进程并与之通信。其次,如果您不需要,分叉子进程是不必要的资源浪费。其他问题包括引用/转义,如果您可能会在代码中引入安全漏洞未能正确清理用户输入。)

例如:

from shutil import copytree
from shutil import Error

try:
   copytree('dir_a', 'dir_b')
except (Error, OSError), e:
    print "Attempt to copy failed: %s" % e

此外,您不应通过将字符串连接在一起来构建文件系统路径,而应使用os.path.join()。这将为当前操作系统使用正确的目录分隔符 ( os.sep),并允许您轻松编写可移植代码。

例子:

>>> import os
>>> os.path.join('/usr/lib', 'python2.7')
'/usr/lib/python2.7'

注意:os.path.join仍然只进行(智能)字符串操作 - 它不关心该路径是否可访问甚至是否存在。

于 2012-09-04T16:36:20.090 回答