当前位置: 首页 > news >正文

Python Cookbook-2.22 计算目录间的相对路径

任务

需要知道一个目录对另一个目录的相对路径是什么–比如,有时需要创建一个符号链接或者一个相对的 URL 引用。

解决方案

最简单的方法是把目录拆分到一个目录的列表中,然后对列表进行处理。我们需要用到一些辅助函数和助手函数,代码如下:

import os,itertools
def all_equal(elements):
'''若所有元素都相等,则返回True,否则返回False'''
	first_element = elements[0]
	for other_element in elements[1:]:
		if other_element != first_element:return False
		return True
def common_prefix(*sequences):
'''返回所有序列开头部分共同元素的列表
紧接一个各序列的不同尾部的列表'''
	#如果没有sequence,完成
	if not sequences: return [ ],[ ]
	#并行的循环序列
	common = [ ]
	for elements in itertools.izip(*sequences):
	#若所有元素相等,跳出循环
	if not all_equal(elements):break
	#得到一个共同的元素,添加到末尾并继续
	common.append(elements[0])
	#返回相同的头部和各自不同的尾部
	return common,[sequence[len(common):] for sequence in sequences]
def relpath(pl,p2,sep=os.path.sep,pardir=os.path.pardir):
'''
返回p1对p2的相对路径
特殊情况:空串,if pl == P2;
p2,如果p2和p1完全没有相同的元素
'''
	common,(u1,u2) = common_prefix(p1.split(sep), p2.split(sep))
	if not common:
		return p2
	return  sep.join([pardir]*len(u1)+u2)
#如果完全没有共同元素,则路径是绝对路径
def test(pl,p2,sep = os.path.sep):
'''调用relpath函数,打印调用参数和结果'''
	print "from",pl,"to",p2,"->",relpath(pl,p2,sep)
if __name__ == '__main__':
	test('/a/b/c/d','/a/b/c1/d1','/')
	test('/a/b/c/d','/a/b/c/d','/')
	test('c:/x/y/z','d:/x/y/z','/')

讨论

本节解决方案给出的代码中,简单而通用的commonprefx是关键部分,给它任意个序列,它能返回 N个序列共同的头部,和一个各不相同的尾部列表。为了计算两个目录之间的相对路径,可以忽略掉它们的共同头部。我们只需要一定数目的“向上一级”标记(通常用 os.path.pardir,比如类UNIX 系统中的…/;需要和尾部长度相同数目的这种符号),然后再添上目标目录的尾部。relpath函数将一个完整路径拆成一个目录的列表,然后调用common_prefix,接着执行我们刚才描述过的操作。

common_prefix的核心部分在那个循环,for elements in itertools.izip(*sequences),它依赖这个事实:当最短的序列循环到头时,izip也结束了。循环的主体部分必须在遇到一个不全相等的元组(根据 izip的说明,每个元素都来自各序列)时立刻结束,同时还要在这个过程中把全等的元素放进common 列表中保存起来。一旦循环结束,剩下要做的事情就是根据 common 列表,把各个序列的相同头部全部切掉。

all_equal 函数还能用另一种方式实现,没那么简洁明快,但是也很有趣:

def all_equal(elements):
	return len(dict.fromkeys(elements)) == 1

或者,等价但更简洁一点,适用于Python 2.4 以上:

def all_equal(elements):
return len(set(elements)) == 1

所有元素相等,等价于包含且只包含这些元素的集合的势(cardinality)为1。在使用dict.fromkeys 的变体中,用 dict 来代替 set,所以那个例子可同时适用于 Python 2.3 和2.4。用 set 的那个例子更清晰,但只能在 Python 2.4 以上的版本中使用(可以用 Python标准库中的 sets模块来改写,让它也同时适用于 Python 2.3)。

相关文章:

  • JS逆向-233网校sid参数和sign参数
  • EfficientViT模型详解及代码复现
  • upload
  • 线程状态与线程方法详解
  • 三支一扶入职体检不合格项目全解析
  • 算法day5 bfs搜索
  • 图像分类项目1:基于卷积神经网络的动物图像分类
  • JavaEE基础之- 过滤器和监听器Filter and Listener
  • 迷你世界脚本状态接口:Buff
  • 在.net中,async/await的理解
  • 【实战篇】【深度解析DeepSeek:从机器学习到深度学习的全场景落地指南】
  • 通往 AI 之路:Python 机器学习入门-面向对象编程
  • 数据库拓展操作
  • 阿里云 Qwen2.5-Max:超大规模 MoE 模型架构和性能评估
  • 大白话面试遇难题,应对策略是什么?
  • 微信小程序开发学习笔记
  • 知识库技术选型:主流Embedding模型特性对比
  • 阿里云ECS Ubuntu PPTP VPN无法访问以太网
  • 使用SPI总线与外部传感器通信,使用ECU抽象
  • 【Git】Ubuntu 安装 Git Large File Storage(LFS)以及使用 Git LFS 下载
  • 克隆网站怎么做/网站优化是什么
  • 如何用.net做网站/百度网盘客服在线咨询
  • 网站建设人才有哪些/aso优化哪家好
  • 网站中信息更新怎么做的/百度网站是什么
  • 成都五月花网页设计培训/北京seo排名优化网站
  • 武汉高端网站建设/站长工具流量统计