用Python将cp -r from_dir/* to_dir

5 投票
3 回答
2856 浏览
提问于 2025-04-16 05:43

有没有简单的方法可以用Python来模拟这个命令 cp -r from_dir/* to_dir 呢?因为 shutil.copytree 不适合,因为 to_dir 已经存在了。

3 个回答

2
import glob
import subprocess

subprocess.check_call(["cp", "-rt", "to_dir"] + glob.glob("from_dir/*"))

有时候,自己在Python里直接做所有事情挺不错的;但有时候,直接调用你知道怎么控制并且有效的命令会更好。

如果需求变化,我会毫不犹豫地重写这段代码,但在那之前,这段代码简短易读——把更多时间花在更大的问题上更有意义。一个需求变化的好例子是报告错误:你没有提到这一点,但一旦需要,我就不会再解析cp的输出了。

2

你只需要用正确的名字(或者相同的名字)来使用 copytree

shutil.copytree("/path/from_dir","/destination/from_dir")
7

看看shutil.copytree的源代码,修改一下,然后使用它:

def copytree(src, dst, symlinks=False, ignore=None):
    """Recursively copy a directory tree using copy2().

    The destination directory must not already exist.
    If exception(s) occur, an Error is raised with a list of reasons.

    If the optional symlinks flag is true, symbolic links in the
    source tree result in symbolic links in the destination tree; if
    it is false, the contents of the files pointed to by symbolic
    links are copied.

    The optional ignore argument is a callable. If given, it
    is called with the `src` parameter, which is the directory
    being visited by copytree(), and `names` which is the list of
    `src` contents, as returned by os.listdir():

        callable(src, names) -> ignored_names

    Since copytree() is called recursively, the callable will be
    called once for each directory that is copied. It returns a
    list of names relative to the `src` directory that should
    not be copied.

    XXX Consider this example code rather than the ultimate tool.

    """
    names = os.listdir(src)
    if ignore is not None:
        ignored_names = ignore(src, names)
    else:
        ignored_names = set()

    os.makedirs(dst)
    errors = []
    for name in names:
        if name in ignored_names:
            continue
        srcname = os.path.join(src, name)
        dstname = os.path.join(dst, name)
        try:
            if symlinks and os.path.islink(srcname):
                linkto = os.readlink(srcname)
                os.symlink(linkto, dstname)
            elif os.path.isdir(srcname):
                copytree(srcname, dstname, symlinks, ignore)
            else:
                copy2(srcname, dstname)
            # XXX What about devices, sockets etc.?
        except (IOError, os.error), why:
            errors.append((srcname, dstname, str(why)))
        # catch the Error from the recursive copytree so that we can
        # continue with other files
        except Error, err:
            errors.extend(err.args[0])
    try:
        copystat(src, dst)
    except OSError, why:
        if WindowsError is not None and isinstance(why, WindowsError):
            # Copying file access times may fail on Windows
            pass
        else:
            errors.extend((src, dst, str(why)))
    if errors:
        raise Error, errors

撰写回答