@@ -195,10 +195,10 @@ def _select_unique(paths):
195
195
yielded = set ()
196
196
try :
197
197
for path in paths :
198
- raw_path = path . _raw_path
199
- if raw_path not in yielded :
198
+ path_str = str ( path )
199
+ if path_str not in yielded :
200
200
yield path
201
- yielded .add (raw_path )
201
+ yielded .add (path_str )
202
202
finally :
203
203
yielded .clear ()
204
204
@@ -247,9 +247,9 @@ class PurePath:
247
247
"""
248
248
249
249
__slots__ = (
250
- # The `_raw_path ` slot stores an unnormalized string path . This is set
250
+ # The `_raw_paths ` slot stores unnormalized string paths . This is set
251
251
# in the `__init__()` method.
252
- '_raw_path ' ,
252
+ '_raw_paths ' ,
253
253
254
254
# The `_drv`, `_root` and `_tail_cached` slots store parsed and
255
255
# normalized parts of the path. They are set when any of the `drive`,
@@ -306,10 +306,11 @@ def __init__(self, *args):
306
306
paths = []
307
307
for arg in args :
308
308
if isinstance (arg , PurePath ):
309
- path = arg ._raw_path
310
309
if arg ._flavour is ntpath and self ._flavour is posixpath :
311
310
# GH-103631: Convert separators for backwards compatibility.
312
- path = path .replace ('\\ ' , '/' )
311
+ paths .extend (path .replace ('\\ ' , '/' ) for path in arg ._raw_paths )
312
+ else :
313
+ paths .extend (arg ._raw_paths )
313
314
else :
314
315
try :
315
316
path = os .fspath (arg )
@@ -320,13 +321,8 @@ def __init__(self, *args):
320
321
"argument should be a str or an os.PathLike "
321
322
"object where __fspath__ returns a str, "
322
323
f"not { type (path ).__name__ !r} " )
323
- paths .append (path )
324
- if len (paths ) == 0 :
325
- self ._raw_path = ''
326
- elif len (paths ) == 1 :
327
- self ._raw_path = paths [0 ]
328
- else :
329
- self ._raw_path = self ._flavour .join (* paths )
324
+ paths .append (path )
325
+ self ._raw_paths = paths
330
326
331
327
def with_segments (self , * pathsegments ):
332
328
"""Construct a new path object from any number of path-like objects.
@@ -356,7 +352,14 @@ def _parse_path(cls, path):
356
352
return drv , root , parsed
357
353
358
354
def _load_parts (self ):
359
- drv , root , tail = self ._parse_path (self ._raw_path )
355
+ paths = self ._raw_paths
356
+ if len (paths ) == 0 :
357
+ path = ''
358
+ elif len (paths ) == 1 :
359
+ path = paths [0 ]
360
+ else :
361
+ path = self ._flavour .join (* paths )
362
+ drv , root , tail = self ._parse_path (path )
360
363
self ._drv = drv
361
364
self ._root = root
362
365
self ._tail_cached = tail
@@ -687,10 +690,17 @@ def parents(self):
687
690
def is_absolute (self ):
688
691
"""True if the path is absolute (has both a root and, if applicable,
689
692
a drive)."""
690
- # ntpath.isabs() is defective - see GH-44626 .
691
693
if self ._flavour is ntpath :
694
+ # ntpath.isabs() is defective - see GH-44626.
692
695
return bool (self .drive and self .root )
693
- return self ._flavour .isabs (self ._raw_path )
696
+ elif self ._flavour is posixpath :
697
+ # Optimization: work with raw paths on POSIX.
698
+ for path in self ._raw_paths :
699
+ if path .startswith ('/' ):
700
+ return True
701
+ return False
702
+ else :
703
+ return self ._flavour .isabs (str (self ))
694
704
695
705
def is_reserved (self ):
696
706
"""Return True if the path contains one of the special names reserved
0 commit comments