我正在尝试使用Python复制文件/目录树。
使用copytree复制以foo结尾的所有内容是否可能?
有一个ignore_patterns模式函数,我可以给它一个否定的正则表达式吗?Python支持它们吗?
例如:
copytree(src,dest,False,ignore_pattern('!*.foo')) 其中!表示不是以foo结尾的任何内容。 谢谢。
我正在尝试使用Python复制文件/目录树。
使用copytree复制以foo结尾的所有内容是否可能?
有一个ignore_patterns模式函数,我可以给它一个否定的正则表达式吗?Python支持它们吗?
例如:
copytree(src,dest,False,ignore_pattern('!*.foo')) 其中!表示不是以foo结尾的任何内容。 谢谢。
shutil.copytree
有一个名为ignore
的关键字参数。可以将ignore
设置为任何可调用对象。给定正在访问的目录和其内容列表,可调用对象应返回要忽略的目录和文件名序列。
例如:
import shutil
def ignored_files(adir,filenames):
return [filename for filename in filenames if not filename.endswith('foo')]
shutil.copytree(source, destination, ignore=ignored_files)
os.path.join(adir, filename)
上调用 os.path.isdir
。 - Steve Rossiter在unutbu的回答基础上进行改进。以下代码获取所有文件的列表,然后移除与"ignore_patterns"匹配的文件,并将其作为应该忽略的文件列表返回。也就是说,它采用了双重否定的方式,只复制您想要的文件。
import glob, os, shutil
def copyonly(dirpath, contents):
return set(contents) - set(
shutil.ignore_patterns('*.py', '*.el')(dirpath, contents),
)
shutil.copytree(
src='.',
dst='temp/',
ignore=copyonly,
)
print glob.glob('temp/*')
def documentation(format):
call(['make', format, '-C', DOC_SOURCE_DIR])
if (os.path.exists(DOC_DIR)):
shutil.rmtree(DOC_DIR)
ignored = ['doctrees']
shutil.copytree('{0}/build/'.format(DOC_SOURCE_DIR), DOC_DIR, ignore=shutil.ignore_patterns(*ignored))
在@unutbu和@johntellsall的基础上构建,这个解决方案也可以递归地工作。
def inverse_ignore_patterns(*patterns: str):
def _inverse_ignore_patterns(path, names):
ignored_names = set(names)
for pattern in patterns:
ignored_names -= set(fnmatch.filter(names, pattern))
# Keep folders
ignored_names -= {
name for name in ignored_names
if os.path.isdir(os.path.join(path, name))
}
return ignored_names
return _inverse_ignore_patterns
一个缺点是,如果目录中没有匹配的文件,它将创建空目录。您可以轻松解决这个问题:
def rm_empty_folders(rootdir: str):
for dirpath, dirnames, filenames in os.walk(rootdir, topdown=False):
if not os.listdir(dirpath):
os.rmdir(dirpath)
使用方法:
copytree(src, dest, False, inverse_ignore_patterns("*.foo"))
rm_empty_folders(dest)