将yield与python中的if / else循环结合起来

问题描述 投票:1回答:3

我想在法语单词列表中加入两个用星号(*)分隔的单词。加入这些单词后,我想检查这个单词是否存在于法语词典中。如果是这样,连接的单词应保留在列表中,否则应将其附加到另一个列表中。我在我的代码中使用了yield(我是这个函数的新手),但我的嵌套if / else循环有问题。谁能帮助我实现目标?我的不成功代码如下:

words = ['Bien', '*', 'venue', 'pour', 'les','engage', '*', 'ment','trop', 'de', 'YIELD', 'peut','être','contre', '*', 'productif' ]

with open ('Fr-dictionary.txt') as fr:
    dic = word_tokenize(fr.read().lower())

l=[ ]

def join_asterisk(ary):
    i, size = 0, len(ary)
    while i < size-2:
        if ary[i+1] == '*':
            if ary[i] + ary[i+2] in dic:
                yield ary[i] + ary[i+2]
                i+=2
            else: yield ary[i]
            i+=1
            l.append(ary[i] + ary[i+2])
    if i < size:
        yield ary[i]



print(list(join_asterisk(words)))
python if-statement while-loop yield
3个回答
1
投票

生成器对于这个用例是完美的,你可以考虑生成器的方式是一个一个接一个地给你产生的值而不是一次性产生的函数(如返回的那样)。换句话说,您可以将其视为不在内存中的列表,只有在被请求时才会获得下一个元素的列表。还要注意发电机只是构建iterators的一种方式。

在你的情况下,这意味着你不必建立一个列表l来跟踪正确的单词,因为生成器join_asterisk会为你产生正确的单词。您需要做的是迭代此生成器将产生的所有值。这正是list(generator)将要做的,它将通过迭代生成器的所有值来构建列表。

最后,代码看起来像这样:

# That look better to me (just in case you change it later)
word_separator = '*'

words = ['Bien', word_separator, 'venue', 'pour', 'les','engage', word_separator, 'ment','trop', 'de', 'YIELD', 'peut', word_separator, "tard"]

# Fake dictionary
dic = {"Bienvenue", "pour", "les", "engagement", "trop", "de", "peut", "peut-être"}

def join_asterisk(ary):
   for w1, w2, w3 in zip(words, words[1:], words[2:]):
      if w2 == word_separator:
        word = w1 + w3
        yield (word, word in dic)
      elif w1 != word_separator and w1 in dic: 
         yield (w1, True)


correct_words = []
incorrect_words = []
for word, is_correct in join_asterisk(words):
  if is_correct:
    correct_words.append(word)
  else:
    incorrect_words.append(word)

print(correct_words)
print(incorrect_words)

这输出

['Bienvenue', 'pour', 'les', 'engagement', 'trop', 'de']
['peuttard']

另请注意,您可以使用列表推导而不是使用for循环来填充两个列表:

correct_words = [w for w, correct in join_asterisk(words) if correct]
incorrect_words = [w for w, correct in join_asterisk(words) if not correct]

0
投票

看起来像线条:

        i+=1
        l.append(ary[i] + ary[i+2])

没有缩进,因此没有参与else。这意味着每个带有* in between的单词将被加到l而不仅仅是不在dic中的对。


0
投票

你不是在寻找这样的东西:

def join_asterisk(ary):
i, size = 0, len(ary)
while i < size-2:
    if ary[i+1] == '*':
        if ary[i] + ary[i+2] in dic:
            yield ary[i] + ary[i+2]
            i+=2
        else: 
            yield ary[i]
            i+=1
        l.append(ary[i] + ary[i+2])
if i < size:
    yield ary[i]

'else'块遵循相同的规则。

例如,在'if','elif','else'或'while'子句的同一行中添加表达式是有效的,但是如果你想要多于on子句的表达式,你必须使用缩进或用';'像这样:

while 1:print(9,end='');print(8)
© www.soinside.com 2019 - 2024. All rights reserved.