Поддерживает ли Python короткое замыкание в логических выражениях?
Переведено автоматически
Ответ 1
Да, оба оператора and и or имеют короткое замыкание - см. Документы.
Ответ 2
Поведение при коротком замыкании в operator and, or:
Давайте сначала определим полезную функцию для определения, выполняется что-либо или нет. Простая функция, которая принимает аргумент, печатает сообщение и возвращает входные данные без изменений.
>>> deffun(i): ... print"executed" ... return i ...
>>> fun(1) executed 1 >>> 1or fun(1) # due to short-circuiting "executed" not printed 1 >>> 1and fun(1) # fun(1) called and "executed" printed executed 1 >>> 0and fun(1) # due to short-circuiting "executed" not printed 0
Примечание: Интерпретатор считает, что следующие значения означают false:
FalseNone0"" () [] {}
Поведение при коротком замыкании в функции: any(), all():
Функции any() и all() Python также поддерживают короткое замыкание. Как показано в документации; они оценивают каждый элемент последовательности по порядку, пока не найдут результат, позволяющий завершить вычисление на раннем этапе. Рассмотрите примеры ниже, чтобы понять оба.
Функция any() проверяет, является ли какой-либо элемент True . Он прекращает выполнение, как только встречается True, и возвращает True .
>>> any(fun(i) for i in [1, 2, 3, 4]) # bool(1) = True executed True >>> any(fun(i) for i in [0, 2, 3, 4]) executed # bool(0) = False executed # bool(2) = True True >>> any(fun(i) for i in [0, 0, 3, 4]) executed executed executed True
Функция all() проверяет, что все элементы истинны, и прекращает выполнение, как только встречается значение False:
>>> all(fun(i) for i in [0, 0, 3, 4]) executed False >>> all(fun(i) for i in [1, 0, 3, 4]) executed executed False
Поведение при коротком замыкании при цепном сравнении:
Кроме того, в Python
Сравнения могут быть скомпонованы произвольно; например, x < y <= z эквивалентно x < y and y <= z, за исключением того, что y вычисляется только один раз (но в обоих случаях z вообще не вычисляется, когда x < y оказывается ложным).
>>> 5 > 6 > fun(3) # same as: 5 > 6 and 6 > fun(3) False# 5 > 6 is False so fun() not called and "executed" NOT printed >>> 5 < 6 > fun(3) # 5 < 6 is True executed # fun(3) called and "executed" printed True >>> 4 <= 6 > fun(7) # 4 <= 6 is True executed # fun(3) called and "executed" printed False >>> 5 < fun(6) < 3# only prints "executed" once executed False >>> 5 < fun(6) and fun(6) < 3# prints "executed" twice, because the second part executes it again executed executed False
Редактировать: Следует отметить еще один интересный момент :- Логические and, or операторы в Python возвращают значение операнда вместо логического значения (True или False). Например:
Операция x and y дает результат if x is false, then x, else y
В отличие от других языков, например, &&, || операторы в C, которые возвращают либо 0, либо 1.
Примеры:
>>> 3and5# Second operand evaluated and returned 5 >>> 3and () () >>> () and5# Second operand NOT evaluated as first operand () is false () # so first operand returned
Аналогично or оператор возвращает самое левое значение, для которого bool(value) == True else самое правое значение false (в соответствии с поведением при коротком замыкании), примеры:
>>> 2or5# left most operand bool(2) == True 2 >>> 0or5# bool(0) == False and bool(5) == True 5 >>> 0or () ()
Итак, чем это полезно? Один из примеров приведен в Практическом Python Магнусом Ли Хетландом:
Допустим, пользователь должен ввести свое имя, но может ничего не вводить, и в этом случае вы хотите использовать значение по умолчанию '<Unknown>'. Вы могли бы использовать оператор if , но вы также могли бы изложить вещи очень кратко:
In [171]: name = raw_input('Enter Name: ') or'<Unknown>' Enter Name:
In [172]: name Out[172]: '<Unknown>'
Другими словами, если возвращаемое значение из raw_input является true (не пустой строкой), оно присваивается name (ничего не меняется); в противном случае по умолчанию '<Unknown>' присваивается name.
Ответ 3
ДА. Попробуйте следующее в вашем интерпретаторе python:
и
>>>Falseand3/0 False >>>Trueand3/0 ZeroDivisionError: integer division or modulo by zero
или
>>>Trueor3/0 True >>>Falseor3/0 ZeroDivisionError: integer division or modulo by zero