本节书摘来自异步社区《Python Cookbook(第2版)中文版》一书中的第1章,第1.7节,作者[美]Alex Martelli , Anna Martelli Ravenscrof , David Ascher ,高铁军 译,更多章节内容可以访问云栖社区“异步社区”公众号查看。
1.7 将字符串逐字符或逐词反转
任务
把字符串逐字符或逐词反转过来。
解决方案
字符串无法改变,所以,反转一个字符串需要创建一个拷贝。最简单的方法是使用一种“步长”为-1的特别的切片方法,这样可立即产生一个完全反转的效果:
revchars = astring[::-1]
如果要按照单词来反转字符串,我们需要先创建一个单词的列表,将这个列表反转,最后再用join方法将其合并,并在相邻两词之间都插入一个空格:
revwords = astring.split( ) # 字符串->单词列表
revwords.reverse( ) # 反转列表
revwords = ' '.join(revwords) # 单词列表->字符串
或者,如果喜欢简练而紧凑的“一行解决”的代码:
revwords = ' '.join(astring.split( )[::-1])
如果想逐词反转但又同时不改变原先的空格,可以用正则表达式来分隔原字符串:
import re
revwords = re.split(r'(\s+)', astring) # 切割字符串为单词列表
revwords.reverse( ) # 反转列表
revwords = ''.join(revwords) # 单词列表 -> 字符串
注意,最后的join操作要使用空字符串,因为空格分隔符已经被保存在revwords列表中了(通过re.split,使用了一个带括弧的组的正则表达式)。当然,也可以写成“一行解决”的形式,只要你乐意:
revwords = ''.join(re.split(r'(\s+)', astring)[::-1])
不过这样显得过于紧凑,也失去了可读性,不是好的Python代码。
讨论
在Python 2.4中,可以改写那个“一行解决”的逐词反转的代码,使用新的内建函数reversed来替代原先的可读性略差的切片指示符[::-1]:
revwords = ' '.join(reversed(astring.split( )))
revwords = ''.join(reversed(re.split(r'(\s+)', astring)))
至于逐字符反转,astring[::-1]仍然是最好的方式,即使在Python 2.4中,因为如果要使用reversed,还得调用".join:
revchars = ''.join(reversed(astring))
新的内建函数reversed返回一个迭代器(iterator),该对象可以被用于循环或者传递给其他的“累加器”,比如".join,但它并不是一个已经完成的字符串。