Python文件操作应优先用pathlib和shutil:pathlib.Path用/运算符替代os.path.join(),自动跨平台并规范化路径;shutil.copy()等函数比os.system("cp")更安全可靠,避免shell依赖和路径错误。
Python 的 os 模块不是“万能文件管家”,它只是对操作系统底层接口的轻量封装;真正稳定、安全、跨平台的文件操作,必须配合 pathlib 使用,且多数场景下应优先用 shutil 而非手动拼接 os.path.join()。
根本问题不在路径分隔符本身,而在于开发者常把字符串拼接逻辑和路径构造混为一谈。比如 os.path.join("a/", "b") 在 Linux 下返回 "a/b",但 os.path.join("a/", "/b") 会直接丢弃前面所有路径,只返回 "/b"(因为遇到绝对路径就重置)。
/(Linux/macOS)或 C:\\(Windows)会让 os.path.join() 忽略之前所有参数"dir/")可能引发意外的嵌套层级,尤其在循环构建路径时os.path.join() 不做路径规范化,os.path.join("a", "..", "b") 返回 "a/../b",而非 "b"
pathlib.Path 把路径当成对象来操作,天然支持运算符重载和链式调用,自动处理平台差异和规范化。
from pathlib import Pathp = Path("a") / "b" / ".." / "c" print(p) # PosixPath('a/b/../c') print(p.resolve()) # PosixPath('/full/path/to/c') —— 真实绝对路径,已规范化 print(p.exists()) # bool,比 os.path.exists(p) 更直观
/ 运算符替代 os.path.join(),语义清晰,无绝对路径截断风险.resolve() 主动展开 ./.. 并检查是否存在,.absolute() 只补前缀不校验Path 对象,不可变,避免误改原始路径变量用 os.system("cp ...") 或 subprocess.run(["cp", ...]) 是典型反模式:依赖 shell、不处理空格路径、无统一错误码、Windows 下直接失效。
shutil.copy(src, dst) 自动处理二进制/文本模式、权限继承、跨文件系统复制shutil.copy2() 额外保留 atime/mtime 时间戳,适合备份场景shutil.copytree() 是唯一安全递归复制目录的方式;os.system("cp -r") 遇到符号链接或特殊权限极易失败shutil.copy() 要求 dst 是完整目标路径(含文件名),不是目录os.listdir() 返回纯文件名列表,不带路径、不递归、不支持通配符;glob.glob() 返回完整路径匹配结果,但默认不递归且不隐藏文件。
os.listdir() + Path(p).joinpath(f)
*.py 或 **/*.py 模式 → 用 glob.glob("*.py", recursive=True) 或更推荐 Path(".").rglob("*.py")
os.scandir() 是高性能替代:一次系统调用获取文件名+类型+stat信息,适合大目录过滤真正难的是权限、编码、符号链接、挂载点边界这些隐性问题——它们不会报错,但会让脚本在某台机器上静默失败。