是否有一种方法可以从Python内部获得类似于shell上的mkdir -p的功能。我正在寻找一个解决方案,而不是系统调用。我确信代码少于20行,我想知道是否有人已经写了它?


当前回答

如果文件已经存在,Mkdir -p会给出一个错误:

$ touch /tmp/foo
$ mkdir -p /tmp/foo
mkdir: cannot create directory `/tmp/foo': File exists

因此,对前面建议的改进是,如果os.path.isdir返回False(在检查errno.EEXIST时),将重新引发异常。

(更新)看看这个高度相似的问题;我同意公认的答案(和注意事项),除了我建议os.path.isdir而不是os.path.exists。

(更新)根据评论中的建议,完整的功能看起来像:

import os
def mkdirp(directory):
    if not os.path.isdir(directory):
        os.makedirs(directory) 

其他回答

这比捕获异常更容易:

import os
if not os.path.exists(...):
    os.makedirs(...)

这种方法需要两次系统调用,在某些环境/条件下更容易受到竞争条件的影响。如果您正在编写比在受控环境中运行的简单的一次性脚本更复杂的东西,那么您最好使用只需要一次系统调用的可接受答案。

更新2012-07-27

我很想删除这个答案,但我认为下面的评论有价值。因此,我将其转换为wiki。

如果文件已经存在,Mkdir -p会给出一个错误:

$ touch /tmp/foo
$ mkdir -p /tmp/foo
mkdir: cannot create directory `/tmp/foo': File exists

因此,对前面建议的改进是,如果os.path.isdir返回False(在检查errno.EEXIST时),将重新引发异常。

(更新)看看这个高度相似的问题;我同意公认的答案(和注意事项),除了我建议os.path.isdir而不是os.path.exists。

(更新)根据评论中的建议,完整的功能看起来像:

import os
def mkdirp(directory):
    if not os.path.isdir(directory):
        os.makedirs(directory) 

最近,我发现这个distutils.dir_util.mkpath:

In [17]: from distutils.dir_util import mkpath

In [18]: mkpath('./foo/bar')
Out[18]: ['foo', 'foo/bar']
import os
import tempfile

path = tempfile.mktemp(dir=path)
os.makedirs(path)
os.rmdir(path)

在Python中>=3.2,这是

os.makedirs(path, exist_ok=True)

在早期版本中,使用@tzot的答案。