Οпростая функция any()

Встроенная функция Python any(iterable) может помочь быстро проверить, есть ли bool(element) True в повторяющемся типе.

>>> l = [None, False, 0]
>>> any(l)
False
>>> l = [None, 1, 0]
>>> any(l)
True

Но есть ли элегантный способ или функция в Python, который мог бы достичь противоположного эффекта any(iterable)? То есть, если какой-либо bool(element) is False, то верните True, как в следующем примере:

>>> l = [True, False, True]
>>> any_false(l)
>>> True

Ответ 1

Существует также функция all, которая делает противоположное тому, что вы хотите, она возвращает True, если все True и False, если они есть False. Поэтому вы можете просто сделать:

not all(l)

Ответ 2

Напишите выражение генератора, которое проверяет ваше пользовательское условие. Вы не обязаны проверять подлинность по умолчанию:

any(not i for i in l)

Ответ 3

Ну, реализация any эквивалентна :

def any(iterable):
    for element in iterable:
        if element:
            return True
    return False

Итак, просто переключите условие с if element на if not element:

def reverse_any(iterable):
    for element in iterable:
        if not element:
            return True
    return False

Да, конечно, это не влияет на скорость встроенных модулей any или all, как это делают другие ответы, но это хорошая читаемая альтернатива.

Ответ 4

Вы можете сделать:

>>> l = [True, False, True]
>>> False in map(bool, l)
True

Напомним, что map в Python 3 является генератором. Для Python 2 вы, вероятно, захотите использовать imap


Mea Culpa: После выбора времени, метод, который я предложил, - это руки вниз самый медленный

Самый быстрый - not all(l) или not next(filterfalse(bool, it), True), который является просто глупым вариантом itertools. Используйте решение Jack Aidleys .

Временной код:

from itertools import filterfalse

def af1(it):
    return not all(it)

def af2(it):
    return any(not i for i in it)   

def af3(iterable):
    for element in iterable:
        if not element:
            return True
    return False    

def af4(it):
    return False in map(bool, it)   

def af5(it):
    return not next(filterfalse(bool, it), True)    

if __name__=='__main__':
    import timeit   
    for i, l in enumerate([[True]*1000+[False]+[True]*999, # False in the middle
                           [False]*2000, # all False
                            [True]*2000], # all True
                            start=1): 
        print("case:", i)
        for f in (af1, af2, af3, af4, af5):
            print("   ",f.__name__, timeit.timeit("f(l)", setup="from __main__ import f, l", number=100000), f(l) )

Результаты:

case: 1
    af1 0.45357259700540453 True
    af2 4.538436588976765 True
    af3 1.2491040650056675 True
    af4 8.935278153978288 True
    af5 0.4685744970047381 True
case: 2
    af1 0.016299808979965746 True
    af2 0.04787631600629538 True
    af3 0.015038023004308343 True
    af4 0.03326922300038859 True
    af5 0.029870904982089996 True
case: 3
    af1 0.8545824179891497 False
    af2 8.786235476000002 False
    af3 2.448748088994762 False
    af4 17.90895140200155 False
    af5 0.9152941330103204 False