一尘不染

Python-循环“忘记”删除某些项目

python

在这段代码中,我试图创建一个函数anti_vowel,该函数将从字符串中删除所有元音(aeiouAEIOU)。我认为它应该可以正常运行,但是当我运行它时,示例文本为“ Hey look Words!”。返回为“ Hy lk Words!”。它“忘记”删除最后一个“ o”。怎么会这样?

text = "Hey look Words!"

def anti_vowel(text):

    textlist = list(text)

    for char in textlist:
        if char.lower() in 'aeiou':
            textlist.remove(char)

    return "".join(textlist)

print anti_vowel(text)

阅读 486

收藏
2020-02-08

共1个答案

一尘不染

你正在修改要遍历的列表,这势必会导致某些不直观的行为。相反,请复制列表,这样就不会从迭代中删除元素。

for char in textlist[:]: #shallow copy of the list
    # etc

要弄清你所看到的行为,请检查一下。放在print char, textlist你的(原始)循环的开头。你可能希望,这会在列表的旁边垂直打印出你的字符串,但是你实际上会得到以下信息:

H ['H', 'e', 'y', ' ', 'l', 'o', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
e ['H', 'e', 'y', ' ', 'l', 'o', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
  ['H', 'y', ' ', 'l', 'o', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!'] # !
l ['H', 'y', ' ', 'l', 'o', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
o ['H', 'y', ' ', 'l', 'o', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
k ['H', 'y', ' ', 'l', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!'] # Problem!!
  ['H', 'y', ' ', 'l', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
W ['H', 'y', ' ', 'l', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
o ['H', 'y', ' ', 'l', 'o', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!'] 
d ['H', 'y', ' ', 'l', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
s ['H', 'y', ' ', 'l', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
! ['H', 'y', ' ', 'l', 'k', ' ', 'W', 'o', 'r', 'd', 's', '!']
Hy lk Words!

发生什么了?for x in yPython中的一个不错的循环实际上只是语法糖:它仍然按索引访问列表元素。因此,当你在列表上进行迭代时从列表中删除元素时,就会开始跳过值(如上所示)。结果,你再也看不到第二个o"look";你可以跳过它,因为删除上一个元素时索引具有高级的“粘贴”它。然后,当你到达oin"Words",你将删除第一次出现的'o',这是你之前跳过的那个。

正如其他人提到的那样,列表理解可能是一种更好(更清晰)的方法。利用Python字符串可迭代的事实:

def remove_vowels(text): # function names should start with verbs! :)
    return ''.join(ch for ch in text if ch.lower() not in 'aeiou')
2020-02-08