在这些条件"built in"下执行 for 循环的 pythonic 方法是什么?



在类方法中,我有以下内容:

if self.zot == 1:
        for foo in self.bar:
            if self.zot != 1:
                break
            #code that may or may not change self.zot

除非绝对必要,否则我宁愿不访问和填充bar属性,因此如果zot不是 1,我仍然希望阻止第一次迭代。

Python就是python,我猜还有一种方法可以做到这一点。

像下面这样?

for foo in self.bar while self.zot == 1:

与python 2.7.x兼容的答案的奖励积分

编辑:在迭代过程中是否修改zot部分独立于bar的内容,因此takewhile是不切实际的。无论哪种方式bar如果不是 1,首先都需要zot不被访问。

您可以将

for转换为while循环并使用iter()next(),尽管它不是很pythonic,并且需要使用try except

iterator = None
while self.zot == 1:
    iterator = iterator or iter(self.bar)
    try:
        current = iterator.next()
        #your code
    except StopIteration:
        break

仅当您第一次进入while时,才会开始使用self.bar

您可以创建一个代理函数来委派检查。

def check_and_iterate(iterable, predicate):
    i = None
    while predicate():
        i = i or iter(iterable)
        yield i.next()
for foo in check_and_iterate(self.bar, lambda: self.zot == 1):
    #your code

编辑:正如@user2357112指出的那样,无论如何都可以访问self.bar。您可以使用另一个 lambda 来推迟它:

def check_and_iterate(getter, predicate):
    i = None
    while predicate():
        i = i or iter(getter())
        yield i.next()
for foo in check_and_iterate(lambda: self.bar, lambda: self.zot == 1):
    #your code

类似的东西,属性bar处理zot检查

@property
def bar(self):
    itr = iter(self._bar)
    while self.zot == 1:
        yield next(itr)

最新更新