34

python 库pathlib提供Path.relative_to. 如果一个路径是另一个路径的子路径,则此函数可以正常工作,如下所示:

from pathlib import Path
foo = Path("C:\\foo")
bar = Path("C:\\foo\\bar")
bar.relative_to(foo)

> WindowsPath('bar')

但是,如果两条路径在同一级别上relative_to则不起作用。

baz = Path("C:\\baz")
foo.relative_to(baz)

> ValueError: 'C:\\foo' does not start with 'C:\\baz'

我希望结果是

WindowsPath("..\\baz")

该函数os.path.relpath正确执行此操作:

import os
foo = "C:\\foo"
bar = "C:\\bar"
os.path.relpath(foo, bar)

> '..\\foo'

有没有办法实现os.path.relpathusing的功能pathlib.Path

4

2 回答 2

34

第一部分解决了 OP 的问题,但如果像我一样,他真的想要相对于公共根的解决方案,那么第二部分会为他解决。第三部分描述了我最初是如何处理它的,并且出于兴趣而保留。

相对路径

最近,与 Python 3.4-6 一样,该os.path模块已扩展为接受pathlib.Path对象。但是,在以下情况下,它不会返回 Path 对象,而是强制包装结果。

foo = Path("C:\\foo")
baz = Path("C:\\baz")
Path(os.path.relpath(foo, baz))

> Path("..\\foo")

共同路径

我的怀疑是你真的在寻找相对于共同根的路径。如果是这种情况,来自EOL的以下内容更有用

Path(os.path.commonpath([foo, baz]))

> Path('c:/root')

通用前缀

在我发现之前,os.path.commonpath我使用过os.path.comonprefix.

foo = Path("C:\\foo")
baz = Path("C:\\baz")
baz.relative_to(os.path.commonprefix([baz,foo]))

> Path('baz')

但请注意,您不应该在这种情况下使用它(参见commonprefix :是的,那个老栗子

foo = Path("C:\\route66\\foo")
baz = Path("C:\\route44\\baz")
baz.relative_to(os.path.commonprefix([baz,foo]))

> ...
> ValueError : `c:\\route44\baz` does not start with `C:\\route`

而是来自JF Sebastian的以下一个。

Path(*os.path.commonprefix([foo.parts, baz.parts]))

> Path('c:/root')

...或者如果您感觉很冗长...

from itertools import takewhile
Path(*[set(i).pop() for i in (takewhile(lambda x : x[0]==x[1], zip(foo.parts, baz.parts)))])
于 2017-04-25T14:38:34.843 回答
5

这让我很烦恼,所以这是一个我认为可以做的仅 pathlib 版本os.path.relpath

def relpath(path_to, path_from):
    path_to = Path(path_to).resolve()
    path_from = Path(path_from).resolve()
    try:
        for p in (*reversed(path_from.parents), path_from):
            head, tail = p, path_to.relative_to(p)
    except ValueError:  # Stop when the paths diverge.
        pass
    return Path('../' * (len(path_from.parents) - len(head.parents))).joinpath(tail)
于 2020-03-13T13:54:53.707 回答