自制图片软件优化关键词排名工具
任务
需要知道一个目录对另一个目录的相对路径是什么–比如,有时需要创建一个符号链接或者一个相对的 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 Falsereturn 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 p2return 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)。