如何使用python执行非递归复制

时间:2018-08-29 10:07:46

标签: python copy shutil non-recursive

我需要使用某些模式复制文件。我需要使用shutil在给定目录上执行非递归副本。我尝试下面给出的此代码执行递归复制。是否有任何选项可以指定以执行非递归复制。

    from fnmatch import fnmatch, filter
    from os.path import isdir, join
    from shutil import copytree, ignore_patterns

    src_directory = r'PATH'
    dst_directory = r'PATH'
    copytree(src_directory, ignored_directory,ignore=ignore_patterns('*.txt'))

3 个答案:

答案 0 :(得分:2)

确保目标目录首先存在。 然后创建所有文件的列表,用glob.glob忽略。然后遍历src目录中的所有文件和目录,并将每个文件复制到dst目录(如果它是文件并且不在ignore列表中):

import os, shutil, glob
src = 'my_dir'
dst = 'my_dir_cp'

try:
    os.mkdir(dst)
except FileExistsError:
    pass

ignore = glob.glob(os.path.join(src, '*.txt'))
for file in os.listdir(src):
    file = os.path.join(src, file)
    if file not in ignore and os.path.isfile(file):
        shutil.copy(file, dst)

完整示例:

$ ls my_dir
bob  cat.txt  fish  jim
$ python -q
>>> import os, shutil, glob
>>> src = 'my_dir'
>>> dst = 'my_dir_cp'
>>> 
>>> try:
...     os.mkdir(dst)
... except FileExistsError:
...     pass
... 
>>> ignore = glob.glob(os.path.join(src, '*.txt'))
>>> for file in os.listdir(src):
...     file = os.path.join(src, file)
...     if file not in ignore and os.path.isfile(file):
...         shutil.copy(file, dst)
... 
'my_dir_cp/jim'
'my_dir_cp/bob'
>>> 
$ ls my_dir_cp
bob  jim

如果您希望能够忽略多个glob模式,则可以glob.glob每个模式并将结果连接在一起(以列表方式):

import os, shutil, glob
src = 'my_dir'
dst = 'my_dir_cp'

try:
    os.mkdir(dst)
except FileExistsError:
    pass

patterns_to_ignore = ['*.txt', '*.bat']
ignore = [e for p in patterns_to_ignore for e in glob.glob(os.path.join(src, p))]
for file in os.listdir(src):
    file = os.path.join(src, file)
    if file not in ignore and os.path.isfile(file):
        shutil.copy(file, dst)

答案 1 :(得分:1)

这更像是黑客,但会以非递归方式将目录中的所有文件复制到目标目录。您可以使用glob或正则表达式忽略文件-

import re

def copytree(src, dst):
     for item in [f for f in os.listdir(src) if os.path.isfile(os.path.join(src, f)) if not re.match(f, '[.]txt$')]:
         s = os.path.join(src, item)
         d = os.path.join(dst, item)
         shutil.copy2(s, d)

答案 2 :(得分:0)

当您选择使用 copytree 中的 shutil 时,请注意任何循环文件以复制每个文件的解决方案都会遗漏 copytree 的许多不错的功能,例如作为符号链接处理、复制文件元数据等。

您可以通过创建一个名为 copytree 的非递归 copyfiles 来保留所有附加项。 copyfilescopytree 的工作原理相同,不同之处在于它扩展了 ignore 参数以在复制时也忽略文件夹。

import shutil, os

def copyfiles(src, dst, ignore=lambda a, b: set(), **kwargs):
    return shutil.copytree(src, dst, ignore=lambda s, names: set(ignore(s, names)).union(set(filter(lambda name: os.path.isdir(os.path.join(s, name)), names))), **kwargs)
# use copyfiles like so:

copyfiles(src_directory, dst_directory, ignore=shutil.ignore_patterns('*.txt')) # copies non-recursively