"pathlib" --- 面向对象的文件系统路径
************************************

Added in version 3.4.

**源代码:** Lib/pathlib/

======================================================================

该模块提供表示文件系统路径的类，其语义适用于不同的操作系统。路径类被分
为提供纯计算操作而没有 I/O 的 纯路径，以及从纯路径继承而来但提供 I/O
操作的 具体路径。

[图片： 继承关系图显示了 pathlib 中所有可用的类。 最基础的类是
PurePath，它有三个直接子类： PurePosixPath, PureWindowsPath 和 Path。
在这四个类之外，还有两个使用多重继承的类： PosixPath 子类
PurePosixPath 和 Path，以及 WindowsPath 子类 PureWindowsPath 和 Path。
][图片]

如果以前从未用过此模块，或不确定哪个类适合完成任务，那要用的可能就是
"Path"。它在运行代码的平台上实例化为 具体路径。

在一些用例中纯路径很有用，例如：

1. 如果你想要在 Unix 设备上操作 Windows 路径（或者相反）。你不应在
   Unix 上实例化一个 "WindowsPath"，但是你可以实例化 "PureWindowsPath"
   。

2. 你只想操作路径但不想实际访问操作系统。在这种情况下，实例化一个纯路
   径是有用的，因为它们没有任何访问操作系统的操作。

参见: **PEP 428**：pathlib 模块 -- 面向对象的文件系统路径。

参见: 对于底层的路径字符串操作，你也可以使用 "os.path" 模块。


基础使用
========

导入主类:

   >>> from pathlib import Path

列出子目录:

   >>> p = Path('.')
   >>> [x for x in p.iterdir() if x.is_dir()]
   [PosixPath('.hg'), PosixPath('docs'), PosixPath('dist'),
    PosixPath('__pycache__'), PosixPath('build')]

列出当前目录树下的所有 Python 源代码文件:

   >>> list(p.glob('**/*.py'))
   [PosixPath('test_pathlib.py'), PosixPath('setup.py'),
    PosixPath('pathlib.py'), PosixPath('docs/conf.py'),
    PosixPath('build/lib/pathlib.py')]

在目录树中移动:

   >>> p = Path('/etc')
   >>> q = p / 'init.d' / 'reboot'
   >>> q
   PosixPath('/etc/init.d/reboot')
   >>> q.resolve()
   PosixPath('/etc/rc.d/init.d/halt')

查询路径的属性:

   >>> q.exists()
   True
   >>> q.is_dir()
   False

打开一个文件:

   >>> with q.open() as f: f.readline()
   ...
   '#!/bin/bash\n'


异常
====

exception pathlib.UnsupportedOperation

   一个继承自 "NotImplementedError" 的异常，当在路径对象上调用不受支持
   的操作时它将被引发。

   Added in version 3.13.


纯路径
======

纯路径对象提供了不实际访问文件系统的路径处理操作。有三种方式来访问这些
类，也是不同的风格：

class pathlib.PurePath(*pathsegments)

   一个通用的类，代表当前系统的路径风格（实例化为 "PurePosixPath" 或者
   "PureWindowsPath"）:

      >>> PurePath('setup.py')      # 在 Unix 机器上运行
      PurePosixPath('setup.py')

   *pathsegments* 的每个元素既可以是代表一个路径段的字符串，也可以是实
   现了 "os.PathLike" 接口的对象，其中 "__fspath__()" 方法返回一个字符
   串，例如另一个路径对象:

      >>> PurePath('foo', 'some/path', 'bar')
      PurePosixPath('foo/some/path/bar')
      >>> PurePath(Path('foo'), Path('bar'))
      PurePosixPath('foo/bar')

   当 *pathsegments* 为空的时候，假定为当前目录:

      >>> PurePath()
      PurePosixPath('.')

   如果某个段为绝对路径，则其前面的所有段都会被忽略 (类似
   "os.path.join()"):

      >>> PurePath('/etc', '/usr', 'lib64')
      PurePosixPath('/usr/lib64')
      >>> PureWindowsPath('c:/Windows', 'd:bar')
      PureWindowsPath('d:bar')

   在 Windows 上，当遇到带根符号的路径段 (如 "r'\foo'") 时驱动器将不会
   被重置:

      >>> PureWindowsPath('c:/Windows', '/Program Files')
      PureWindowsPath('c:/Program Files')

   假斜杠和单个点号会被消除，但双点号 ("'..'") 和打头的双斜杠 ("'//'")
   不会，因为这会出于各种原因改变路径的实际含义 (例如符号链接、UNC 路
   径等):

      >>> PurePath('foo//bar')
      PurePosixPath('foo/bar')
      >>> PurePath('//foo/bar')
      PurePosixPath('//foo/bar')
      >>> PurePath('foo/./bar')
      PurePosixPath('foo/bar')
      >>> PurePath('foo/../bar')
      PurePosixPath('foo/../bar')

   （一个很 naïve 的做法是让 "PurePosixPath('foo/../bar')" 等同于
   "PurePosixPath('bar')"，如果 "foo" 是一个指向其他目录的符号链接那么
   这个做法就将出错）

   纯路径对象实现了 "os.PathLike" 接口，允许它们在任何接受此接口的地方
   使用。

   在 3.6 版本发生变更: 添加了 "os.PathLike" 接口支持。

class pathlib.PurePosixPath(*pathsegments)

   一个 "PurePath" 的子类，路径风格不同于 Windows 文件系统:

      >>> PurePosixPath('/etc/hosts')
      PurePosixPath('/etc/hosts')

   *pathsegments* 参数的指定和 "PurePath" 相同。

class pathlib.PureWindowsPath(*pathsegments)

   "PurePath" 的一个子类，此路径风格代表 Windows 文件系统路径，包括
   UNC paths:

      >>> PureWindowsPath('c:/', 'Users', 'Ximénez')
      PureWindowsPath('c:/Users/Ximénez')
      >>> PureWindowsPath('//server/share/file')
      PureWindowsPath('//server/share/file')

   *pathsegments* 参数的指定和 "PurePath" 相同。

无论你正运行什么系统，你都可以实例化这些类，因为它们提供的操作不做任何
系统调用。


通用性质
--------

路径是不可变并且 *hashable*。 相同风格的路径可以排序和比较。 这此特性
会尊重对应风格的大小写转换语义。:

   >>> PurePosixPath('foo') == PurePosixPath('FOO')
   False
   >>> PureWindowsPath('foo') == PureWindowsPath('FOO')
   True
   >>> PureWindowsPath('FOO') in { PureWindowsPath('foo') }
   True
   >>> PureWindowsPath('C:') < PureWindowsPath('d:')
   True

不同风格的路径比较得到不等的结果并且无法被排序:

   >>> PureWindowsPath('foo') == PurePosixPath('foo')
   False
   >>> PureWindowsPath('foo') < PurePosixPath('foo')
   Traceback (most recent call last):
     File "<stdin>", line 1, in <module>
   TypeError: '<' not supported between instances of 'PureWindowsPath' and 'PurePosixPath'


运算符
------

斜杠操作符可以帮助创建子路径，如 "os.path.join()"。 如果参数为一个绝对
路径，则之前的路径会被忽略。 在 Windows 上，当参数为一个带根符号的相对
路径 (如 "r'\foo'") 时驱动器将不会被重置:

   >>> p = PurePath('/etc')
   >>> p
   PurePosixPath('/etc')
   >>> p / 'init.d' / 'apache2'
   PurePosixPath('/etc/init.d/apache2')
   >>> q = PurePath('bin')
   >>> '/usr' / q
   PurePosixPath('/usr/bin')
   >>> p / '/an_absolute_path'
   PurePosixPath('/an_absolute_path')
   >>> PureWindowsPath('c:/Windows', '/Program Files')
   PureWindowsPath('c:/Program Files')

文件对象可用于任何接受 "os.PathLike" 接口实现的地方。

   >>> import os
   >>> p = PurePath('/etc')
   >>> os.fspath(p)
   '/etc'

路径的字符串表示法为它自己原始的文件系统路径（以原生形式，例如在
Windows 下使用反斜杠）。你可以传递给任何需要字符串形式路径的函数。

   >>> p = PurePath('/etc')
   >>> str(p)
   '/etc'
   >>> p = PureWindowsPath('c:/Program Files')
   >>> str(p)
   'c:\\Program Files'

类似地，在路径上调用 "bytes" 将原始文件系统路径作为字节对象给出，就像
被 "os.fsencode()" 编码一样:

   >>> bytes(p)
   b'/etc'

备注:

  只推荐在 Unix 下调用 "bytes"。在 Windows， unicode 形式是文件系统路
  径的规范表示法。


访问个别部分
------------

为了访问路径独立的部分 （组件），使用以下特征属性：

PurePath.parts

   一个元组，可以访问路径的多个组件:

      >>> p = PurePath('/usr/bin/python3')
      >>> p.parts
      ('/', 'usr', 'bin', 'python3')

      >>> p = PureWindowsPath('c:/Program Files/PSF')
      >>> p.parts
      ('c:\\', 'Program Files', 'PSF')

   （注意盘符和本地根目录是如何重组的）


方法和特征属性
--------------

纯路径提供以下方法和特征属性：

PurePath.parser

   用于底层路径解析与合并的 "os.path" 模块的实现: "posixpath" 或
   "ntpath"。

   Added in version 3.13.

PurePath.drive

   一个表示驱动器盘符或命名的字符串，如果存在:

      >>> PureWindowsPath('c:/Program Files/').drive
      'c:'
      >>> PureWindowsPath('/Program Files/').drive
      ''
      >>> PurePosixPath('/etc').drive
      ''

   UNC 分享也被认作驱动器:

      >>> PureWindowsPath('//host/share/foo.txt').drive
      '\\\\host\\share'

PurePath.root

   一个表示（本地或全局）根的字符串，如果存在:

      >>> PureWindowsPath('c:/Program Files/').root
      '\\'
      >>> PureWindowsPath('c:Program Files/').root
      ''
      >>> PurePosixPath('/etc').root
      '/'

   UNC 分享一样拥有根:

      >>> PureWindowsPath('//host/share').root
      '\\'

   如果路径以超过两个连续斜框打头，"PurePosixPath" 会合并它们:

      >>> PurePosixPath('//etc').root
      '//'
      >>> PurePosixPath('///etc').root
      '/'
      >>> PurePosixPath('////etc').root
      '/'

   备注:

     此行为符合 *The Open Group Base Specifications Issue 6*,
     paragraph 4.11 Pathname Resolution:*"以连续两个斜杠打头的路径名可
     能会以具体实现所定义的方式被解读，但是两个以上的前缀斜杠则应当被
     当作一个斜杠来处理。"*

PurePath.anchor

   驱动器和根的联合:

      >>> PureWindowsPath('c:/Program Files/').anchor
      'c:\\'
      >>> PureWindowsPath('c:Program Files/').anchor
      'c:'
      >>> PurePosixPath('/etc').anchor
      '/'
      >>> PureWindowsPath('//host/share').anchor
      '\\\\host\\share\\'

PurePath.parents

   提供访问此路径的逻辑祖先的不可变序列:

      >>> p = PureWindowsPath('c:/foo/bar/setup.py')
      >>> p.parents[0]
      PureWindowsPath('c:/foo/bar')
      >>> p.parents[1]
      PureWindowsPath('c:/foo')
      >>> p.parents[2]
      PureWindowsPath('c:/')

   在 3.10 版本发生变更: parents 序列现在支持 *切片* 和负的索引值。

PurePath.parent

   此路径的逻辑父路径:

      >>> p = PurePosixPath('/a/b/c/d')
      >>> p.parent
      PurePosixPath('/a/b/c')

   你不能超过一个 anchor 或空路径:

      >>> p = PurePosixPath('/')
      >>> p.parent
      PurePosixPath('/')
      >>> p = PurePosixPath('.')
      >>> p.parent
      PurePosixPath('.')

   备注:

     这是一个单纯的词法操作，因此有以下行为:

        >>> p = PurePosixPath('foo/..')
        >>> p.parent
        PurePosixPath('foo')

     如果你想要向上遍历任意文件系统路径，建议首先调用 "Path.resolve()"
     以便解析符号链接并消除 "".."" 部分。

PurePath.name

   一个表示最后路径组件的字符串，排除了驱动器与根目录，如果存在的话:

      >>> PurePosixPath('my/library/setup.py').name
      'setup.py'

   UNC 驱动器名不被考虑:

      >>> PureWindowsPath('//some/share/setup.py').name
      'setup.py'
      >>> PureWindowsPath('//some/share').name
      ''

PurePath.suffix

   最后一个路径组件中以点分割的后一部分（如果有）

      >>> PurePosixPath('my/library/setup.py').suffix
      '.py'
      >>> PurePosixPath('my/library.tar.gz').suffix
      '.gz'
      >>> PurePosixPath('my/library').suffix
      ''

   这通常被称作文件扩展名。

   在 3.14 版本发生变更: 一个单独点号 (""."") 会被视为有效的前缀。

PurePath.suffixes

   由路径后缀组成的列表，经常被称作文件扩展名:

      >>> PurePosixPath('my/library.tar.gar').suffixes
      ['.tar', '.gar']
      >>> PurePosixPath('my/library.tar.gz').suffixes
      ['.tar', '.gz']
      >>> PurePosixPath('my/library').suffixes
      []

   在 3.14 版本发生变更: 一个单独点号 (""."") 会被视为有效的前缀。

PurePath.stem

   最后一个路径组件，除去后缀:

      >>> PurePosixPath('my/library.tar.gz').stem
      'library.tar'
      >>> PurePosixPath('my/library.tar').stem
      'library'
      >>> PurePosixPath('my/library').stem
      'library'

PurePath.as_posix()

   返回使用正斜杠（"/"）的路径字符串:

      >>> p = PureWindowsPath('c:\\windows')
      >>> str(p)
      'c:\\windows'
      >>> p.as_posix()
      'c:/windows'

PurePath.is_absolute()

   返回此路径是否为绝对路径。如果路径同时拥有驱动器符与根路径（如果风
   格允许）则将被认作绝对路径。

      >>> PurePosixPath('/a/b').is_absolute()
      True
      >>> PurePosixPath('a/b').is_absolute()
      False

      >>> PureWindowsPath('c:/a/b').is_absolute()
      True
      >>> PureWindowsPath('/a/b').is_absolute()
      False
      >>> PureWindowsPath('c:').is_absolute()
      False
      >>> PureWindowsPath('//some/share').is_absolute()
      True

PurePath.is_relative_to(other)

   返回此路径是否相对于 *other* 的路径。

   >>> p = PurePath('/etc/passwd')
   >>> p.is_relative_to('/etc')
   True
   >>> p.is_relative_to('/usr')
   False

   此方法是基于字符串的；它不会访问文件系统也不会对 "".."" 部分进行特
   殊处理。 以下代码是等价的：

   >>> u = PurePath('/usr')
   >>> u == p or u in p.parents
   False

   Added in version 3.9.

   从 3.12 版起已弃用，已在 3.14 版中移除: 传入附加参数的做法已被弃用
   ；如果提供了附加参数，它们将与 *other* 合并。

PurePath.is_reserved()

   在 "PureWindowsPath"，如果路径是被 Windows 保留的则返回 "True"，否
   则 "False"。在 "PurePosixPath"，总是返回 "False"。

   在 3.13 版本发生变更: Windows 的含有一个冒号或以点或空格结尾的路径
   名是被认为是被保留的。 UNC 路径也可能被保留。

   从 3.13 版起已弃用，将在 3.15 版中移除: 此方法已被弃用；使用
   "os.path.isreserved()" 以鉴别在 Windows 下的保留路径。

PurePath.joinpath(*pathsegments)

   调用此方法等同于依次将路径与给定的每个 *pathsegments* 组合到一起:

      >>> PurePosixPath('/etc').joinpath('passwd')
      PurePosixPath('/etc/passwd')
      >>> PurePosixPath('/etc').joinpath(PurePosixPath('passwd'))
      PurePosixPath('/etc/passwd')
      >>> PurePosixPath('/etc').joinpath('init.d', 'apache2')
      PurePosixPath('/etc/init.d/apache2')
      >>> PureWindowsPath('c:').joinpath('/Program Files')
      PureWindowsPath('c:/Program Files')

PurePath.full_match(pattern, *, case_sensitive=None)

   将此路径与所提供的 glob 样式匹配。如果匹配成功，则返回 "True"，否则
   返回 "False"。 例如:

      >>> PurePath('a/b.py').full_match('a/*.py')
      True
      >>> PurePath('a/b.py').full_match('*.py')
      False
      >>> PurePath('/a/b/c.py').full_match('/a/**')
      True
      >>> PurePath('/a/b/c.py').full_match('**/*.py')
      True

   参见: 模式语言 文档。

   与其他方法一样，是否大小写敏感遵循平台的默认规则:

      >>> PurePosixPath('b.py').full_match('*.PY')
      False
      >>> PureWindowsPath('b.py').full_match('*.PY')
      True

   将 *case_sensitive* 设为 "True" 或 "False" 来覆盖此行为。

   Added in version 3.13.

PurePath.match(pattern, *, case_sensitive=None)

   将此路径与所提供的非递归 glob 样式匹配。 如果匹配成功，则返回
   "True"，否则返回 "False"。

   此方法与 "full_match()" 类似，但不允许空模式 (将引发 "ValueError")
   ，也不支持递归通配符 ""**"" (将视为非递归的 ""*"")，并且如果提供了
   一个相对格式，则将从右开始匹配:

      >>> PurePath('a/b.py').match('*.py')
      True
      >>> PurePath('/a/b/c.py').match('b/*.py')
      True
      >>> PurePath('/a/b/c.py').match('a/*.py')
      False

   在 3.12 版本发生变更: *pattern* 形参接受一个 *path-like object*。

   在 3.12 版本发生变更: 增加了 *case_sensitive* 形参。

PurePath.relative_to(other, walk_up=False)

   计算此路径相对于 *other* 所表示路径的版本。 如果不可计算，则引发
   "ValueError":

      >>> p = PurePosixPath('/etc/passwd')
      >>> p.relative_to('/')
      PurePosixPath('etc/passwd')
      >>> p.relative_to('/etc')
      PurePosixPath('passwd')
      >>> p.relative_to('/usr')
      Traceback (most recent call last):
        File "<stdin>", line 1, in <module>
        File "pathlib.py", line 941, in relative_to
          raise ValueError(error_message.format(str(self), str(formatted)))
      ValueError: '/etc/passwd' is not in the subpath of '/usr' OR one path is relative and the other is absolute.

   当 *walk_up* 为（默认的）假值时，路径必须以 *other* 开始。 当参数为
   真值时，可能会添加 ".." 条目以形成相对路径。 在所有其他情况下，例如
   路径引用了不同的驱动器，则会引发 "ValueError"。:

      >>> p.relative_to('/usr', walk_up=True)
      PurePosixPath('../etc/passwd')
      >>> p.relative_to('foo', walk_up=True)
      Traceback (most recent call last):
        File "<stdin>", line 1, in <module>
        File "pathlib.py", line 941, in relative_to
          raise ValueError(error_message.format(str(self), str(formatted)))
      ValueError: '/etc/passwd' is not on the same drive as 'foo' OR one path is relative and the other is absolute.

   警告:

     该函数是 "PurePath" 的一部分并适用于字符串。 它不会检查或访问下层
     的文件结构体。 这可能会影响 *walk_up* 选项因为它假定路径中不存在
     符号链接；如果需要处理符号链接请先调用 "resolve()"。

   在 3.12 版本发生变更: 增加了 *walk_up* 形参数（原本的行为与
   "walk_up=False" 相同）。

   从 3.12 版起已弃用，已在 3.14 版中移除: 传入附加位置参数的做法已被
   弃用；如果提供的话，它们将与 *other* 合并。

PurePath.with_name(name)

   返回一个新的路径并修改 "name"。如果原本路径没有 name，ValueError 被
   抛出:

      >>> p = PureWindowsPath('c:/Downloads/pathlib.tar.gz')
      >>> p.with_name('setup.py')
      PureWindowsPath('c:/Downloads/setup.py')
      >>> p = PureWindowsPath('c:/')
      >>> p.with_name('setup.py')
      Traceback (most recent call last):
        File "<stdin>", line 1, in <module>
        File "/home/antoine/cpython/default/Lib/pathlib.py", line 751, in with_name
          raise ValueError("%r has an empty name" % (self,))
      ValueError: PureWindowsPath('c:/') has an empty name

PurePath.with_stem(stem)

   返回一个带有修改后 "stem" 的新路径。 如果原路径没有名称，则会引发
   ValueError:

      >>> p = PureWindowsPath('c:/Downloads/draft.txt')
      >>> p.with_stem('final')
      PureWindowsPath('c:/Downloads/final.txt')
      >>> p = PureWindowsPath('c:/Downloads/pathlib.tar.gz')
      >>> p.with_stem('lib')
      PureWindowsPath('c:/Downloads/lib.gz')
      >>> p = PureWindowsPath('c:/')
      >>> p.with_stem('')
      Traceback (most recent call last):
        File "<stdin>", line 1, in <module>
        File "/home/antoine/cpython/default/Lib/pathlib.py", line 861, in with_stem
          return self.with_name(stem + self.suffix)
        File "/home/antoine/cpython/default/Lib/pathlib.py", line 851, in with_name
          raise ValueError("%r has an empty name" % (self,))
      ValueError: PureWindowsPath('c:/') has an empty name

   Added in version 3.9.

PurePath.with_suffix(suffix)

   返回一个新的路径并修改 "suffix"。如果原本的路径没有后缀，新的
   *suffix* 则被追加以代替。如果 *suffix* 是空字符串，则原本的后缀被移
   除:

      >>> p = PureWindowsPath('c:/Downloads/pathlib.tar.gz')
      >>> p.with_suffix('.bz2')
      PureWindowsPath('c:/Downloads/pathlib.tar.bz2')
      >>> p = PureWindowsPath('README')
      >>> p.with_suffix('.txt')
      PureWindowsPath('README.txt')
      >>> p = PureWindowsPath('README.txt')
      >>> p.with_suffix('')
      PureWindowsPath('README')

   在 3.14 版本发生变更: 单个点号 (""."") 会被视为合法的后缀。 在之前
   版本中，如果提供单个点号则会引发 "ValueError"。

PurePath.with_segments(*pathsegments)

   通过组合给定的 *pathsegments* 创建一个相同类型的新路径对象。 每当创
   建派生路径，如从 "parent" 和 "relative_to()" 创建时都会调用此方法。
   子类可以覆盖此方法以便向派生路径传递信息，例如:

      from pathlib import PurePosixPath

      class MyPath(PurePosixPath):
          def __init__(self, *pathsegments, session_id):
              super().__init__(*pathsegments)
              self.session_id = session_id

          def with_segments(self, *pathsegments):
              return type(self)(*pathsegments, session_id=self.session_id)

      etc = MyPath('/etc', session_id=42)
      hosts = etc / 'hosts'
      print(hosts.session_id)  # 42

   Added in version 3.12.


具体路径
========

具体路径是纯路径的子类。除了后者提供的操作之外，它们还提供了对路径对象
进行系统调用的方法。有三种方法可以实例化具体路径:

class pathlib.Path(*pathsegments)

   一个 "PurePath" 的子类，此类以当前系统的路径风格表示路径（实例化为
   "PosixPath" 或 "WindowsPath"）:

      >>> Path('setup.py')
      PosixPath('setup.py')

   *pathsegments* 参数的指定和 "PurePath" 相同。

class pathlib.PosixPath(*pathsegments)

   一个 "Path" 和 "PurePosixPath" 的子类，此类表示一个非 Windows 文件
   系统的具体路径:

      >>> PosixPath('/etc/hosts')
      PosixPath('/etc/hosts')

   *pathsegments* 参数的指定和 "PurePath" 相同。

   在 3.13 版本发生变更: 在 Windows 上引发 "UnsupportedOperation"。 在
   之前版本中，则为引发 "NotImplementedError"。

class pathlib.WindowsPath(*pathsegments)

   "Path" 和 "PureWindowsPath" 的子类，从类表示一个 Windows 文件系统的
   具体路径:

      >>> WindowsPath('c:/', 'Users', 'Ximénez')
      WindowsPath('c:/Users/Ximénez')

   *pathsegments* 参数的指定和 "PurePath" 相同。

   在 3.13 版本发生变更: 在非 Windows 平台上引发
   "UnsupportedOperation"。 在之前版本中，则为引发
   "NotImplementedError"。

你只能实例化与当前系统风格相同的类（允许系统调用作用于不兼容的路径风格
可能在应用程序中导致缺陷或失败）:

   >>> import os
   >>> os.name
   'posix'
   >>> Path('setup.py')
   PosixPath('setup.py')
   >>> PosixPath('setup.py')
   PosixPath('setup.py')
   >>> WindowsPath('setup.py')
   Traceback (most recent call last):
     File "<stdin>", line 1, in <module>
     File "pathlib.py", line 798, in __new__
       % (cls.__name__,))
   UnsupportedOperation: cannot instantiate 'WindowsPath' on your system

某些具体路径方法在一个系统调用失败时（例如由于路径不存在）可能引发
"OSError"。


解析和生成 URI
--------------

具体路径对象可基于符合 **RFC 8089** 的 '文件' URI 来创建，并可用它来表
示。

备注:

  文件 URI 不能在具有不同 文件系统编码格式 的机器之间进行移植。

classmethod Path.from_uri(uri)

   通过解析一个 '文件' URI 来返回新的路径对象。 例如:

      >>> p = Path.from_uri('file:///etc/hosts')
      PosixPath('/etc/hosts')

   在 Windows 上，可以基于 URI 来解析 DOS 设备和 UNC 路径:

      >>> p = Path.from_uri('file:///c:/windows')
      WindowsPath('c:/windows')
      >>> p = Path.from_uri('file://server/share')
      WindowsPath('//server/share')

   某些变化形式也是受支持的:

      >>> p = Path.from_uri('file:////server/share')
      WindowsPath('//server/share')
      >>> p = Path.from_uri('file://///server/share')
      WindowsPath('//server/share')
      >>> p = Path.from_uri('file:c:/windows')
      WindowsPath('c:/windows')
      >>> p = Path.from_uri('file:/c|/windows')
      WindowsPath('c:/windows')

   如果 URI 不是以 "file:" 开头，或者被解析的不是绝对路径则会引发
   "ValueError"。

   Added in version 3.13.

   在 3.14 版本发生变更: 如果与本地主机名匹配，则丢弃 URL 权限。 否则
   ，如果权限不是空的或 "localhost"，那么在 Windows 上返回 UNC 路径（
   和以前一样），在其他平台上引发 "ValueError"。

Path.as_uri()

   将路径表示为 'file' URI。 如果路径不是绝对路径则会引发 "ValueError"
   。

      >>> p = PosixPath('/etc/passwd')
      >>> p.as_uri()
      'file:///etc/passwd'
      >>> p = WindowsPath('c:/Windows')
      >>> p.as_uri()
      'file:///c:/Windows'

   从 3.14 版起已弃用，将在 3.19 版中移除: 可以从 "PurePath" 而不是
   "Path" 调用此方法，但已弃用。该方法使用 "os.fsencode()" 使其严格不
   纯。


扩展和计算路径
--------------

classmethod Path.home()

   返回一个表示用户家目录的新路径对象（与带 "~" 构造的
   "os.path.expanduser()" 所返回的相同）。 如果无法解析家目录，则会引
   发 "RuntimeError"。

      >>> Path.home()
      PosixPath('/home/antoine')

   Added in version 3.5.

Path.expanduser()

   返回带有扩展 "~" 和 "~user" 构造的新路径，与 "os.path.expanduser()"
   所返回的相同。 如果无法解析家目录，则会引发 "RuntimeError"。

      >>> p = PosixPath('~/films/Monty Python')
      >>> p.expanduser()
      PosixPath('/home/eric/films/Monty Python')

   Added in version 3.5.

classmethod Path.cwd()

   返回一个新的表示当前目录的路径对象（和 "os.getcwd()" 返回的相同）:

      >>> Path.cwd()
      PosixPath('/home/antoine/pathlib')

Path.absolute()

   改为绝对路径，不会执行正规化或解析符号链接。 返回一个新的路径对象:

      >>> p = Path('tests')
      >>> p
      PosixPath('tests')
      >>> p.absolute()
      PosixPath('/home/antoine/pathlib/tests')

Path.resolve(strict=False)

   将路径绝对化，解析任何符号链接。返回新的路径对象:

      >>> p = Path()
      >>> p
      PosixPath('.')
      >>> p.resolve()
      PosixPath('/home/antoine/pathlib')

   "".."" 组件也将被消除（只有这一种方法这么做）:

      >>> p = Path('docs/../setup.py')
      >>> p.resolve()
      PosixPath('/home/antoine/pathlib/setup.py')

   如果一个路径不存在或是遇到了符号链接循环，并且 *strict* 为 "True"，
   则会引发 "OSError"。 如果 *strict* 为 "False"，则会尽可能地解析路径
   并添加任何剩余部分而不会检查其是否存在。

   在 3.6 版本发生变更: 增加了 *strict* 参数（3.6 之前的行为是默认采用
   strict 模式）。

   在 3.13 版本发生变更: 符号链接循环将像其他错误一样处理：在严格模式
   下会引发 "OSError"，而在非严格模式下不会引发异常。 在之前版本中，无
   论 *strict* 的值是什么都会引发 "RuntimeError"。

Path.readlink()

   返回符号链接所指向的路径（即 "os.readlink()" 的返回值）:

      >>> p = Path('mylink')
      >>> p.symlink_to('setup.py')
      >>> p.readlink()
      PosixPath('setup.py')

   Added in version 3.9.

   在 3.13 版本发生变更: 如果 "os.readlink()" 不可用则会引发
   "UnsupportedOperation"。 在之前版本中，则是引发
   "NotImplementedError"。


查询文件类型和状态
------------------

在 3.8 版本发生变更: 现在对于包含在 OS 层级上无法表示的字符的路径
"exists()", "is_dir()", "is_file()", "is_mount()", "is_symlink()",
"is_block_device()", "is_char_device()", "is_fifo()", "is_socket()" 将
返回 "False" 而不是引发异常。

在 3.14 版本发生变更: 上面给出的方法现在返回 "False" 而不是从操作系统
引发任何 "OSError" 异常。 在以前的版本中，会引发某些类型的 "OSError"
异常，而抑制其他类型的异常。 新的行为与 "os.path.exists()",
"os.path.isdir()" 等一致。 使用 "stat()" 来在不抑制异常的情况下检索文
件状态。

Path.stat(*, follow_symlinks=True)

   返回一个 "os.stat_result" 对象，其中包含有关此路径的信息，就像
   "os.stat()"。 结果会在每次调用此方法时被查找。

   此方法通常会跟随符号链接；要对 symlink 使用 stat 请添加参数
   "follow_symlinks=False"，或者使用 "lstat()"。

      >>> p = Path('setup.py')
      >>> p.stat().st_size
      956
      >>> p.stat().st_mtime
      1327883547.852554

   在 3.10 版本发生变更: 增加了 *follow_symlinks* 形参。

Path.lstat()

   就和 "Path.stat()" 一样，但是如果路径指向符号链接，则是返回符号链接
   而不是目标的信息。

Path.exists(*, follow_symlinks=True)

   如果路径指向已存在的文件或目录，则返回 "True"。 如果路径无效、不可
   访问或缺失，将返回 "False"。 使用 "Path.stat()" 来区分这些情况。

   此方法通常会跟随符号链接；要检查符号链接是否存在，请添加参数
   "follow_symlinks=False"。

      >>> Path('.').exists()
      True
      >>> Path('setup.py').exists()
      True
      >>> Path('/etc').exists()
      True
      >>> Path('nonexistentfile').exists()
      False

   在 3.12 版本发生变更: 增加了 *follow_symlinks* 形参。

Path.is_file(*, follow_symlinks=True)

   如果路径指向一个常规文件，则返回 "True"。 如果路径无效、不可访问或
   缺失，或如果它指向的不是一个常规文件，将返回 "False"。 使用
   "Path.stat()" 来区分这些情况。

   此方法通常会跟随符号链接；要排除符号链接，请添加参数
   "follow_symlinks=False"。

   在 3.13 版本发生变更: 增加了 *follow_symlinks* 形参。

Path.is_dir(*, follow_symlinks=True)

   如果路径指向一个目录，则返回 "True"。 如果路径无效、不可访问或缺失
   ，或如果它指向的不是一个目录，将返回 "False"。 使用 "Path.stat()"
   来区分这些情况。

   此方法通常会跟随符号链接；要排除指向目录的符号链接，请添加参数
   "follow_symlinks=False"。

   在 3.13 版本发生变更: 增加了 *follow_symlinks* 形参。

Path.is_symlink()

   如果路径指向一个符号链接，则返回 "True"，即使该符号链接已损坏。 如
   果路径无效、不可访问或缺失，或如果它指向的不是一个符号链接，将返回
   "False"。 使用 "Path.stat()" 来区分这些情况。

Path.is_junction()

   如果路径是指向一个接合点则返回 "True"，如果是其他文件类型则返回
   "False"。 目前只有 Windows 支持接合点。

   Added in version 3.12.

Path.is_mount()

   如果路径是一个 *挂载点*: 在文件系统中被其他不同文件系统挂载的位置则
   返回 "True"。 在 POSIX 上，此函数将检查 *path* 的上一级 "path/.."
   是否位于和 *path* 不同的设备中，或者 "path/.." 和 *path* 是否指向位
   于相同设置的相同 i-node --- 这应当能检测所有 Unix 和 POSIX 变种上的
   挂载点。 在 Windows 上，挂载点是被视为驱动器盘符的根目录 (例如
   "c:\")、UNC 共享目录 (例如 "\\server\share") 或已挂载的文件系统目录
   。

   Added in version 3.7.

   在 3.12 版本发生变更: 添加了 Windows 支持。

Path.is_socket()

   如果路径指向一个 Unix 套接字，则返回 "True"。 如果路径无效、不可访
   问或缺失，或如果它指向的不是一个 Unix 套接字，将返回 "False"。 使用
   "Path.stat()" 来区分这些情况。

Path.is_fifo()

   如果路径指向一个 FIFO（先进先出），则返回 "True"。 如果路径无效、不
   可访问或缺失，或如果它指向的不是一个 FIFO，将返回 "False"。 使用
   "Path.stat()" 来区分这些情况。

Path.is_block_device()

   如果路径指向一个块设备，则返回 "True"。 如果路径无效、不可访问或缺
   失，或如果它指向的不是一个块设备，将返回 "False"。 使用
   "Path.stat()" 来区分这些情况。

Path.is_char_device()

   如果路径指向一个字符设备，则返回 "True"。 如果路径无效、不可访问或
   缺失，或如果它指向的不是一个字符设备，将返回 "False"。 使用
   "Path.stat()" 来区分这些情况。

Path.samefile(other_path)

   返回此目录是否指向与可能是字符串或者另一个路径对象的 *other_path*
   相同的文件。语义类似于 "os.path.samefile()" 与 "os.path.samestat()"
   。

   如果两者都以同一原因无法访问，则抛出 "OSError"。

      >>> p = Path('spam')
      >>> q = Path('eggs')
      >>> p.samefile(q)
      False
      >>> p.samefile('spam')
      True

   Added in version 3.5.

Path.info

   支持查询文件类型信息的 "PathInfo" 对象。 该对象公开了缓存其结果的方
   法，这有助于在切换文件类型时减少所需的系统调用数量。 例如:

      >>> p = Path('src')
      >>> if p.info.is_symlink():
      ...     print('symlink')
      ... elif p.info.is_dir():
      ...     print('directory')
      ... elif p.info.exists():
      ...     print('something else')
      ... else:
      ...     print('not found')
      ...
      directory

   如果路径是由 "Path.iterdir()" 生成的，那么这个属性会初始化一些通过
   扫描父目录收集到的关于文件类型的信息。 仅仅访问 "Path.info" 不会执
   行任何文件系统查询。

   要获取最新的信息，最好调用 "Path.is_dir()"、"is_file()" 和
   "is_symlink()"，而不是调用该属性的方法。 没有办法重置缓存；相反，您
   可以通过 "p = Path(p)" 创建一个空信息缓存的新路径对象。

   Added in version 3.14.


读写文件
--------

Path.open(mode='r', buffering=-1, encoding=None, errors=None, newline=None)

   打开路径指向的文件，就像内置的 "open()" 函数所做的一样:

      >>> p = Path('setup.py')
      >>> with p.open() as f:
      ...     f.readline()
      ...
      '#!/usr/bin/env python3\n'

Path.read_text(encoding=None, errors=None, newline=None)

   以字符串形式返回路径指向的文件的解码后文本内容。

      >>> p = Path('my_text_file')
      >>> p.write_text('Text file contents')
      18
      >>> p.read_text()
      'Text file contents'

   文件先被打开然后关闭。有和 "open()" 一样的可选形参。

   Added in version 3.5.

   在 3.13 版本发生变更: 增加了 *newline* 形参。

Path.read_bytes()

   以字节对象的形式返回路径指向的文件的二进制内容:

      >>> p = Path('my_binary_file')
      >>> p.write_bytes(b'Binary file contents')
      20
      >>> p.read_bytes()
      b'Binary file contents'

   Added in version 3.5.

Path.write_text(data, encoding=None, errors=None, newline=None)

   将文件以文本模式打开，写入 *data* 并关闭:

      >>> p = Path('my_text_file')
      >>> p.write_text('Text file contents')
      18
      >>> p.read_text()
      'Text file contents'

   同名的现有文件会被覆盖。 可选形参的含义与 "open()" 的相同。

   Added in version 3.5.

   在 3.10 版本发生变更: 增加了 *newline* 形参。

Path.write_bytes(data)

   将文件以二进制模式打开，写入 *data* 并关闭:

      >>> p = Path('my_binary_file')
      >>> p.write_bytes(b'Binary file contents')
      20
      >>> p.read_bytes()
      b'Binary file contents'

   一个同名的现存文件将被覆盖。

   Added in version 3.5.


读取目录
--------

Path.iterdir()

   当路径指向一个目录时，产生该路径下的对象的路径:

      >>> p = Path('docs')
      >>> for child in p.iterdir(): child
      ...
      PosixPath('docs/conf.py')
      PosixPath('docs/_templates')
      PosixPath('docs/make.bat')
      PosixPath('docs/index.rst')
      PosixPath('docs/_build')
      PosixPath('docs/_static')
      PosixPath('docs/Makefile')

   子条目会以任意顺序产生，并且不包括特殊条目 "'.'" 和 "'..'"。 如果迭
   代器创建之后有文件在目录中被移除或添加，是否要包括该文件所对应的路
   径对象并没有明确规定。

   如果该路径不是一个目录或是无法访问，则会引发 "OSError"。

Path.glob(pattern, *, case_sensitive=None, recurse_symlinks=False)

   解析相对于此路径的通配符 *pattern*，产生所有匹配的文件:

      >>> sorted(Path('.').glob('*.py'))
      [PosixPath('pathlib.py'), PosixPath('setup.py'), PosixPath('test_pathlib.py')]
      >>> sorted(Path('.').glob('*/*.py'))
      [PosixPath('docs/conf.py')]
      >>> sorted(Path('.').glob('**/*.py'))
      [PosixPath('build/lib/pathlib.py'),
       PosixPath('docs/conf.py'),
       PosixPath('pathlib.py'),
       PosixPath('setup.py'),
       PosixPath('test_pathlib.py')]

   备注:

     路径不会以特定的顺序返回。 如果你需要特定的顺序，请结结果进行排序
     。

   参见: 模式语言 文档。

   在默认情况下，或当 *case_sensitive* 关键字参数被设为 "None" 时，该
   方法将使用特定平台的大小写规则匹配路径：通常，在 POSIX 上区分大小写
   ，而在 Windows 上不区分大小写。将 *case_sensitive* 设为 "True" 或
   "False" 可覆盖此行为。

   在默认情况下，或是当 *recurse_symlinks* 关键字参数被设为 "False" 时
   ，此方法将跟随符号链接但在扩展 ""**"" 通配符时除外。 将
   *recurse_symlinks* 设为 "True" 将总是跟随符号链接。

   引发一个 审计事件 "pathlib.Path.glob" 并附带参数 "self", "pattern"
   。

   在 3.12 版本发生变更: 增加了 *case_sensitive* 形参。

   在 3.13 版本发生变更: 增加了 *recurse_symlinks* 形参。

   在 3.13 版本发生变更: *pattern* 形参接受一个 *path-like object*。

   在 3.13 版本发生变更: 任何因扫描文件系统而引发的 "OSError" 异常都会
   被抑制。 在之前的版本中，此类异常在许多情况下都会被抑制，但并非全部
   情况。

Path.rglob(pattern, *, case_sensitive=None, recurse_symlinks=False)

   递归地对给定的相对 *pattern* 执行 glob 通配。 这类似于调用
   "Path.glob()" 时在 *pattern* 之前加上 ""**/""。

   备注:

     路径不会以特定的顺序返回。 如果你需要特定的顺序，请结结果进行排序
     。

   参见: 模式语言 和 "Path.glob()" 文档。

   引发一个 审计事件 "pathlib.Path.rglob" 并附带参数 "self", "pattern"
   。

   在 3.12 版本发生变更: 增加了 *case_sensitive* 形参。

   在 3.13 版本发生变更: 增加了 *recurse_symlinks* 形参。

   在 3.13 版本发生变更: *pattern* 形参接受一个 *path-like object*。

Path.walk(top_down=True, on_error=None, follow_symlinks=False)

   通过对目录树自上而下或自下而上的遍历来生成其中的文件名。

   对于根位置为 *self* 的目录树中的每个目录（包括 *self* 但不包括 '.'
   和 '..'），该方法会产生一个 3 元组 "(dirpath, dirnames, filenames)"
   。

   *dirpath* 是指向当前正被遍历到的目录的 "Path"，*dirnames* 是由表示
   *dirpath* 中子目录名称的字符串组成的列表 (不包括 "'.'" 和 "'..'")，
   *filenames* 是由表示 *dirpath* 中非目录文件名称的字符串组成的列表。
   要获取 *dirpath* 中文件或目录的完整路径 (以 *self* 开头)，可使用
   "dirpath / name"。 这些列表是否排序取决于具体的文件系统。

   如果可选参数 *top_down* 为（默认的）真值，则会在所有子目录的三元组
   生成之前生成父目录的三元组（目录是自上而下遍历的）。 如果
   *top_down* 为假值，则会在所有子目录的三元组生成之后再生成父目录的三
   元组（目录是是自下而上遍历的）。无论 *top_down* 的值是什么，都会在
   遍历目录及其子目录的三元组之前提取子目录列表。

   当 *top_down* 为真值时，调用方可以原地修改 *dirnames* 列表（例如，
   使用 "del" 或切片赋值），而 "Path.walk()" 只会向名称保留在
   *dirnames* 中的子目录递归。 这可被用于搜索剪枝，或强制应用特定的访
   问顺序，或者甚至是在重新恢复执行 "Path.walk()" 之前告知
   "Path.walk()" 调用方所创建或重命名的目录。 当 *top_down* 为假值时修
   改 *dirnames* 不会对 "Path.walk()" 的的行为造成影响，因为在
   *dirnames* 被提供给调用方时 *dirnames* 中的目录已经被生成了。

   在默认情况下，来自 "os.scandir()" 的错误将被忽略。 如果指定了可选参
   数 *on_error*，则它应为一个可调用对象；调用它需要传入一个参数，即
   "OSError" 的实例。 该可调用对象能处理错误以继续执行遍历或是重新引发
   错误以停止遍历。 请注意可以通过异常对象的 "filename" 属性来获取文件
   名。

   在默认情况下，"Path.walk()" 不会跟踪符号链接，而是将其添加到
   *filenames* 列表中。 将 *follow_symlinks*设为真值可解析符号链接并根
   据它们的目标将其放入*dirnames* 和 *filenames* 中，从而（在受支持的
   系统上）访问符号链接所指向的目录。

   备注:

     请注意将 *follow_symlinks* 设为真值会在链接指向自身的父目录时导致
     无限递归。 "Path.walk()" 不会跟踪已它访问过的目录。

   备注:

     "Path.walk()" 会假定在执行过程中它所遍历的目录没有被修改。 例如，
     如果 *dirnames* 中的某个目录已被替换为符号链接并且
     *follow_symlinks* 为假值，则 "Path.walk()" 仍会尝试进入该目录。
     为防止出现这种行为，请相应地移除 *dirnames* 中的目录。

   备注:

     与 "os.walk()" 不同，当 *follow_symlinks* 为假值时 "Path.walk()"
     会在 *filenames* 中列出指向目录的符号链接。

   这个例子显示每个目录中所有文件使用的总字节数，忽略其中的
   "__pycache__" 目录:

      from pathlib import Path
      for root, dirs, files in Path("cpython/Lib/concurrent").walk(on_error=print):
        print(
            root,
            "consumes",
            sum((root / file).stat().st_size for file in files),
            "bytes in",
            len(files),
            "non-directory files"
        )
        if '__pycache__' in dirs:
              dirs.remove('__pycache__')

   下一个例子是 "shutil.rmtree()" 的一个简单实现。 由于 "rmdir()" 不允
   许在目录为空之前删除该目录因此自下而上地遍历目录树是至关重要的:

      # 删除可从目录 "top" 进入的所有东西。
      # 小心：这很危险！举例来说，如果 top == Path('/')，
      # 它可能会删除你的全部文件。
      for root, dirs, files in top.walk(top_down=False):
          for name in files:
              (root / name).unlink()
          for name in dirs:
              (root / name).rmdir()

   Added in version 3.12.


创建文件和目录
--------------

Path.touch(mode=0o666, exist_ok=True)

   使用给定的路径创建文件。 如果给出了 *mode*，它将与进程的 "umask" 值
   合并以确定文件模式和访问旗标。 如果文件已存在，则当 *exist_ok* 为真
   值时函数将成功执行（并且其修改时间将更新为当前时间），在其他情况下
   则会引发 "FileExistsError"。

   参见: "open()", "write_text()" 和 "write_bytes()" 方法经常被用来创建文
       件。

Path.mkdir(mode=0o777, parents=False, exist_ok=False)

   使用给定的路径新建目录。 如果给出了 *mode*，它将与进程的 "umask" 值
   合并来决定文件模式和访问旗标。 如果路径已存在，则会引发
   "FileExistsError"。

   如果 *parents* 为真值，任何找不到的父目录都会伴随着此路径被创建；它
   们会以默认权限被创建，而不考虑 *mode* 设置（模仿 POSIX 的 "mkdir
   -p" 命令）。

   如果 *parents* 为假值（默认），则找不到的父级目录会引发
   "FileNotFoundError"。

   如果 *exist_ok* 为 false（默认），则在目标已存在的情况下抛出
   "FileExistsError"。

   如果 *exist_ok* 为真值，则 "FileExistsError" 将不会被引发除非给定的
   路径在文件系统中已存在并且不是目录（与 POSIX "mkdir -p" 命令的行为
   相同）。

   在 3.5 版本发生变更: *exist_ok* 形参被加入。

Path.symlink_to(target, target_is_directory=False)

   使该路径成为一个指向 *target* 的符号连接。

   在 Windows，符号链接可以代表文件或者目录，并且不会动态适应目标。 如
   果目标存在，则将创建相匹配的符号链接类型。 在其他情况下，如果
   *target_is_directory* 为真值则符号链接将创建为目录类型否则将创建为
   文件符号链接。 在非 Windows 平台上，*target_is_directory* 将被忽略
   。

      >>> p = Path('mylink')
      >>> p.symlink_to('setup.py')
      >>> p.resolve()
      PosixPath('/home/antoine/pathlib/setup.py')
      >>> p.stat().st_size
      956
      >>> p.lstat().st_size
      8

   备注:

     参数的顺序（link, target) 和 "os.symlink()" 是相反的。

   在 3.13 版本发生变更: 如果 "os.symlink()" 不可用则会引发
   "UnsupportedOperation"。 在之前版本中，则是引发
   "NotImplementedError"。

Path.hardlink_to(target)

   将此路径设为一个指向与 *target* 相同文件的硬链接。

   备注:

     参数顺序 (link, target) 和 "os.link()" 是相反的。

   Added in version 3.10.

   在 3.13 版本发生变更: 如果 "os.link()" 不可用则会引发
   "UnsupportedOperation"。 在之前版本中，则是引发
   "NotImplementedError"。


拷贝、移动和删除
----------------

Path.copy(target, *, follow_symlinks=True, preserve_metadata=False)

   将此文件或目录树拷贝到给定的 *target*，并返回一个指向 *target* 的新
   的 "Path" 实例。

   如果源文件是一个文件，则如果目标文件是一个已存在的文件，则目标文件
   将被替换。如果源是一个符号链接，并且 *follow_symlinks* 为true（默认
   值），则复制该符号链接的目标。否则，将在目的地重新创建符号链接。

   如果 *preserve_metadata* 为false（默认值），则只保证复制目录结构和
   文件数据。将 *preserve_metadata* 设置为true，以确保在支持的地方复制
   文件和目录权限、标志、最后访问和修改时间以及扩展属性。此参数在
   Windows上复制文件时不起作用（会始终保留元数据）。

   备注:

     在操作系统和文件系统支持的情况下，此方法执行轻量级复制，数据块仅
     在被修改时才会被复制。这就是所谓的写时复制。

   Added in version 3.14.

Path.copy_into(target_dir, *, follow_symlinks=True, preserve_metadata=False)

   将这个文件或目录树复制到给定的 *target_dir* 中，它应该是一个现有的
   目录。 其他参数的处理方式与 "Path.copy()" 相同。 返回一个新的指向副
   本的 "Path" 实例。

   Added in version 3.14.

Path.rename(target)

   将此文件或目录重命名为给定的 *target*，并返回一个新的指向 *target*
   的 "Path" 实例。 在 Unix 上，如果 *target* 存在且为一个文件，那么如
   果用户具有相应权限则它将静默地替换。 在 Windows 上，如果 *target*
   存在，则会引发 "FileExistsError"。 *target* 可以是一个字符串或者另
   一个路径对象:

      >>> p = Path('foo')
      >>> p.open('w').write('some text')
      9
      >>> target = Path('bar')
      >>> p.rename(target)
      PosixPath('bar')
      >>> target.open().read()
      'some text'

   目标路径可能为绝对或相对路径。 相对路径将被解读为相对于当前工作目录
   ，而 *不是* 相对于 "Path" 对象的目录。

   它根据 "os.rename()" 实现并给出了同样的保证。

   在 3.8 版本发生变更: 添加了返回值，将返回新的 "Path" 实例。

Path.replace(target)

   将此文件或目录重命名为给定的 *target*，并返回一个新的指向 *target*
   的 "Path" 实例。 如果 *target* 指向一个现有文件或空目录，则它将被无
   条件地替换。

   目标路径可能为绝对或相对路径。 相对路径将被解读为相对于当前工作目录
   ，而 *不是* 相对于 "Path" 对象的目录。

   在 3.8 版本发生变更: 添加了返回值，将返回新的 "Path" 实例。

Path.move(target)

   将此文件或目录树移动到给定的 *target*，并返回一个指向 *target* 的新
   的 "Path" 实例。

   如果 *target* 不存在，它将被创建。 如果该路径和 *target* 都是现有文
   件，则覆盖目标。 如果两个路径都指向相同的文件或目录，或者 *target*
   是非空目录，则引发 "OSError"。

   如果两个路径都在同一个文件系统上，则使用 "os.replace()" 执行移动。
   否则，该路径将被复制（保留元数据和符号链接），再被删除。

   Added in version 3.14.

Path.move_into(target_dir)

   将这个文件或目录树移动到给定的 *target_dir* 中，它应该是一个现有的
   目录。 返回一个新的指向移动后路径的 "Path" 实例。

   Added in version 3.14.

Path.unlink(missing_ok=False)

   移除此文件或符号链接。如果路径指向目录，则用 "Path.rmdir()" 代替。

   如果 *missing_ok* 为假值（默认），则如果路径不存在将会引发
   "FileNotFoundError"。

   如果 *missing_ok* 为真值，则 "FileNotFoundError" 异常将被忽略（和
   POSIX "rm -f" 命令的行为相同）。

   在 3.8 版本发生变更: 增加了 *missing_ok* 形参。

Path.rmdir()

   移除此目录。此目录必须为空的。


访问权限与所有权
----------------

Path.owner(*, follow_symlinks=True)

   返拥有此文件的用户名。 如果文件的用户标识 (UID) 无法在系统数据库中
   找到则会引发 "KeyError"。

   此方法通常会跟随符号链接；要获取符号链接的所有者，请添加参数
   "follow_symlinks=False"。

   在 3.13 版本发生变更: 如果 "pwd" 模块不可用则会引发
   "UnsupportedOperation"。 在之前的版本中，则是引发
   "NotImplementedError"。

   在 3.13 版本发生变更: 增加了 *follow_symlinks* 形参。

Path.group(*, follow_symlinks=True)

   返回拥有此文件的用户组名。 如果文件的用户组标识 (GID) 无法在系统数
   据库中找到则会引发 "KeyError"。

   此方法通常会跟随符号链接；要获取符号链接的用户组，请添加参数
   "follow_symlinks=False"。

   在 3.13 版本发生变更: 如果 "grp" 模块不可用则会引发
   "UnsupportedOperation"。 在较早的版本中，则是引发
   "NotImplementedError"。

   在 3.13 版本发生变更: 增加了 *follow_symlinks* 形参。

Path.chmod(mode, *, follow_symlinks=True)

   改变文件模式和权限，和 "os.chmod()" 一样。

   此方法通常会跟随符号链接。 某些 Unix 变种支持改变 symlink 本身的权
   限；在这些平台上你可以添加参数 "follow_symlinks=False"，或者使用
   "lchmod()"。

      >>> p = Path('setup.py')
      >>> p.stat().st_mode
      33277
      >>> p.chmod(0o444)
      >>> p.stat().st_mode
      33060

   在 3.10 版本发生变更: 增加了 *follow_symlinks* 形参。

Path.lchmod(mode)

   就像 "Path.chmod()" 但是如果路径指向符号链接则是修改符号链接的模式
   ，而不是修改符号链接的目标。


模式语言
========

以下通配符在用于 "full_match()", "glob()" 和 "rglob()" 的模式中是受支
持的：

"**" (整个分段)
   匹配任意数量的文件或目录分段，包括零个。

"*" (整个分段)
   匹配一个文件或目录分段。

"*" (分段的一部分)
   匹配任意数量的非分隔符型字符，包括零个。

"?"
   匹配一个不是分隔符的字符。

"[seq]"
   匹配 *seq* 中的一个字符，*seq* 是一个字符序列。 支持区间表达式；例
   如，"[a-z]" 匹配任意小写 ASCII 字母。 多个区间可以合并：如
   "[a-zA-Z0-9_]" 匹配任意 ASCII 字母、数字或下划线。

"[!seq]"
   匹配不在 *seq* 中的一个字符，*seq* 遵循与上一通配符相同的规则。

对于字面值匹配，请将元字符用方括号括起来。 例如，""[?]"" 将匹配字符
""?""。

""**"" 通配符将启用递归 glob。 下面是几个例子：

+---------------------------+-----------------------------------------------------------------------+
| 模式                      | 含意                                                                  |
|===========================|=======================================================================|
| ""**/*""                  | 任何具有至少一个分段的路径。                                          |
+---------------------------+-----------------------------------------------------------------------+
| ""**/*.py""               | 最后部分以 "".py"" 结尾的任意路径。                                   |
+---------------------------+-----------------------------------------------------------------------+
| ""assets/**""             | 以 ""assets/"" 开头的任意路径。                                       |
+---------------------------+-----------------------------------------------------------------------+
| ""assets/**/*""           | 以 ""assets/"" 开头，但不包括 ""assets/"" 本身的任意路径。            |
+---------------------------+-----------------------------------------------------------------------+

备注:

  使用 ""**"" 通配符执行 glob 操作将访问目录树中的每个目录。 非常大的
  目录树可能要花费非常长的时间来搜索。

在 3.13 版本发生变更: 使用以 ""**"" 结尾的模式执行 glob 操作将同时返回
文件和目录。 在之前的版本中，只有目录会被返回。

在 "Path.glob()" 和 "rglob()" 中，可以向模式添加一个末尾斜杠以只匹配目
录。

在 3.11 版本发生变更: 使用以一个路径名称组件分隔符 ("sep" or "altsep")
结尾的模式执行 glob 操作将只返回目录。


与 "glob" 模块的比较
====================

"Path.glob()" 和 "Path.rglob()" 所接受的模式和所生成的结果相比 "glob"
模式的略有不同：

1. 以点号打点的文件在 pathlib 中没有特殊含义。 这类似于向
   "glob.glob()" 传入 "include_hidden=True"。

2. ""**"" 模式组件在 pathlib 总是递归的。 这类似于向 "glob.glob()" 传
   入 "recursive=True"。

3. ""**"" 模式组件在 pathlib 中默认不会跟随符号链接。 此行为在
   "glob.glob()" 中没有对应物，但你可以向 "Path.glob()" 传入
   "recurse_symlinks=True" 以获得兼容的行为。

4. 与所有 "PurePath" 和 "Path" 对象类似，从 "Path.glob()" 和
   "Path.rglob()" 返回的值都不包括末尾斜杠。

5. 从 pathlib 的 "path.glob()" 和 "path.rglob()" 返回的值包括作为前缀
   的 *path*，这不同于 "glob.glob(root_dir=path)" 的结果。

6. 从 pathlib 的 "path.glob()" 和 "path.rglob()" 返回的值可能包括
   *path* 本身，例如当对 ""**"" 执行 glob 操作的时候，而
   "glob.glob(root_dir=path)" 的结果绝不会包括与 *path* 对应的空字符串
   。


与 "os" 和 "os.path" 模块的比较
===============================

pathlib 使用 "PurePath" 和 "Path" 对象来实现路径操作，因此它被认为是 *
面向对象的*。 而在另一方面，"os" 和 "os.path" 模块提供与低层级 "str"
和 "bytes" 对象配合使用的函数，它更接近于 *面向过程的* 方式。 某些用户
认为面向对象的风格可读性更好。

"os" 和 "os.path" 中的许多函数都支持 "bytes" 路径和 相对于目录描述符的
路径。 这些特性在 pathlib 中均不可用。

Python 的 "str" 和 "bytes" 类型，以及 "os" 和 "os.path" 模块的各个部分
都是用 C 编写的因而非常快速。 pathlib 是用纯 Python 编写的因而往往较慢
，但很少会慢到引人注意。

pathlib 的路径规范化比 "os.path" 更有主见也更一致。 例如，
"os.path.abspath()" 会删除路径中的 "".."" 段，因为当涉及符号链接时这可
能会改变其含义；而 "Path.absolute()" 则会保留这些段以提高安全性。

pathlib 的路径规范化可能使其不适合某些应用程序：

1. pathlib 会将 "Path("my_folder/")" 规范化为 "Path("my_folder")"，这
   改变了路径在提供给各种操作系统 API 和命令行工具时的含义。 具体来说
   ，缺少末尾分隔符可能会使路径被解析为文件或目录，而非只是目录。

2. pathlib 会将 "Path("./my_program")" 规范化为 "Path("my_program")"，
   这改变了路径在被用作可执行文件搜索路径，例如在 shell 中或者在产生子
   进程时的含义。 具体来说，缺少路径中的分隔符可能会迫使其在 "PATH" 而
   不是在当前目录中查找。

由于这些差异的影响，pathlib 并不是 "os.path" 的直接替代品。


相关工具
--------

以下是一个映射了 "os" 与 "PurePath"/"Path" 对应相同的函数的表。

+---------------------------------------+------------------------------------------------+
| "os" 和 "os.path"                     | "pathlib"                                      |
|=======================================|================================================|
| "os.path.dirname()"                   | "PurePath.parent"                              |
+---------------------------------------+------------------------------------------------+
| "os.path.basename()"                  | "PurePath.name"                                |
+---------------------------------------+------------------------------------------------+
| "os.path.splitext()"                  | "PurePath.stem", "PurePath.suffix"             |
+---------------------------------------+------------------------------------------------+
| "os.path.join()"                      | "PurePath.joinpath()"                          |
+---------------------------------------+------------------------------------------------+
| "os.path.isabs()"                     | "PurePath.is_absolute()"                       |
+---------------------------------------+------------------------------------------------+
| "os.path.relpath()"                   | "PurePath.relative_to()" [1]                   |
+---------------------------------------+------------------------------------------------+
| "os.path.expanduser()"                | "Path.expanduser()" [2]                        |
+---------------------------------------+------------------------------------------------+
| "os.path.realpath()"                  | "Path.resolve()"                               |
+---------------------------------------+------------------------------------------------+
| "os.path.abspath()"                   | "Path.absolute()" [3]                          |
+---------------------------------------+------------------------------------------------+
| "os.path.exists()"                    | "Path.exists()"                                |
+---------------------------------------+------------------------------------------------+
| "os.path.isfile()"                    | "Path.is_file()"                               |
+---------------------------------------+------------------------------------------------+
| "os.path.isdir()"                     | "Path.is_dir()"                                |
+---------------------------------------+------------------------------------------------+
| "os.path.islink()"                    | "Path.is_symlink()"                            |
+---------------------------------------+------------------------------------------------+
| "os.path.isjunction()"                | "Path.is_junction()"                           |
+---------------------------------------+------------------------------------------------+
| "os.path.ismount()"                   | "Path.is_mount()"                              |
+---------------------------------------+------------------------------------------------+
| "os.path.samefile()"                  | "Path.samefile()"                              |
+---------------------------------------+------------------------------------------------+
| "os.getcwd()"                         | "Path.cwd()"                                   |
+---------------------------------------+------------------------------------------------+
| "os.stat()"                           | "Path.stat()"                                  |
+---------------------------------------+------------------------------------------------+
| "os.lstat()"                          | "Path.lstat()"                                 |
+---------------------------------------+------------------------------------------------+
| "os.listdir()"                        | "Path.iterdir()"                               |
+---------------------------------------+------------------------------------------------+
| "os.walk()"                           | "Path.walk()" [4]                              |
+---------------------------------------+------------------------------------------------+
| "os.mkdir()", "os.makedirs()"         | "Path.mkdir()"                                 |
+---------------------------------------+------------------------------------------------+
| "os.link()"                           | "Path.hardlink_to()"                           |
+---------------------------------------+------------------------------------------------+
| "os.symlink()"                        | "Path.symlink_to()"                            |
+---------------------------------------+------------------------------------------------+
| "os.readlink()"                       | "Path.readlink()"                              |
+---------------------------------------+------------------------------------------------+
| "os.rename()"                         | "Path.rename()"                                |
+---------------------------------------+------------------------------------------------+
| "os.replace()"                        | "Path.replace()"                               |
+---------------------------------------+------------------------------------------------+
| "os.remove()", "os.unlink()"          | "Path.unlink()"                                |
+---------------------------------------+------------------------------------------------+
| "os.rmdir()"                          | "Path.rmdir()"                                 |
+---------------------------------------+------------------------------------------------+
| "os.chmod()"                          | "Path.chmod()"                                 |
+---------------------------------------+------------------------------------------------+
| "os.lchmod()"                         | "Path.lchmod()"                                |
+---------------------------------------+------------------------------------------------+

-[ 备注 ]-

[1] "os.path.relpath()" 会调用 "abspath()" 以使路径变为绝对路径并移除
    "".."" 部分，而 "PurePath.relative_to()" 是一个词法操作，当其输入
    的锚点不同时（例如当一个路径为绝对路径而另一个为相对路径时）将引发
    "ValueError"。

[2] 若无法解析家目录， "os.path.expanduser()" 会原封不动地返回输入的路
    径，但 "Path.expanduser()" 会抛出 "RuntimeError" 异常。

[3] "os.path.abspath()" 会移除 "".."" 组件而不解析符号链接，这可能改变
    路径的含义，而 "Path.absolute()" 则会让路径中的任何 "".."" 组件保
    持原样。

[4] "os.walk()" 在将路径分类为 *dirnames* 和 *filenames* 时总是会跟随
    符号链接，而 "Path.walk()" 当 *follow_symlinks* 为（默认的）假值时
    会将符号链接分类为 *filenames*。


协议
====

"pathlib.types" 模块提供了用于静态类型检查的类型。

Added in version 3.14.

class pathlib.types.PathInfo

   描述 "Path.info" 属性的 "typing.Protocol"。 各种实现可以从它们的方
   法返回缓存的结果。

   exists(*, follow_symlinks=True)

      如果路径是一个已存在的文件或目录，或任何其他类型的文件，则返回
      ``True``；如果路径不存在，返回``False``。

      如果 *follow_symlinks* 为``False`` ，则对符号链接返回``True``，
      而不检查符号链接的目标是否存在。

   is_dir(*, follow_symlinks=True)

      如果路径是一个目录，或是一个指向目录的符号链接，则返回 "True"。
      如果路径是（或指向）任何其他类型的文件，或者路径不存在，则返回
      "False"。

      如果 *follow_symlinks* 是 "False"，那么仅当该路径为目录时返回
      "True" （不跟踪符号链接）；如果该路径是任何类型的文件，或该路径
      不存在，则返回 "False"。

   is_file(*, follow_symlinks=True)

      如果该路径是文件，或是指向一个文件的符号链接，则返回 "True"。如
      果该路径是（或指向）一个目录或其他非文件，或该路径不存在，则返回
      "False"。

      如果 *follow_symlinks* 是 "False"，仅当该路径是一个文件时返回
      "True" （不跟踪符号链接）；如果该路径是一个目录或其他非文件，或
      该路径不存在，则返回 "False"。

   is_symlink()

      如果该路径是符号链接（即使已断开）则返回 "True"；如果该路径是目
      录或任何种类的文件，或者已不存在则返回 "False"。
