Неожиданное поведение генератора, если он не назначен переменной

Может ли кто-нибудь объяснить разницу между этими двумя казнями? Вот моя функция генератора:

def g(n):
    try:
        yield n
        print("first")
    except BaseException as e:
        print(f"exception {e}")
        raise e
    finally:
        print("second")

Когда я выполняю:

>>> a = next(g(2))
exception
second

Может ли кто-нибудь объяснить, почему a = next(g(2)) вызывает исключение и в чем разница с выполнением ниже?

>>> x = g(2)
>>> next(x)
2

Я ожидал, что когда я выполню a = next(g(2)), функция g(2) вернет генератор, а next() вернет первый выход генератора, но, видимо, возникнет исключение. Почему это происходит?

Почему в Python есть оператор "pass"?
Почему в Python есть оператор "pass"?
Оператор pass в Python - это простая концепция, которую могут быстро освоить даже новички без опыта программирования.
Некоторые методы, о которых вы не знали, что они существуют в Python
Некоторые методы, о которых вы не знали, что они существуют в Python
Python - самый известный и самый простой в изучении язык в наши дни. Имея широкий спектр применения в области машинного обучения, Data Science,...
Основы Python Часть I
Основы Python Часть I
Вы когда-нибудь задумывались, почему в программах на Python вы видите приведенный ниже код?
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
Алиса и Боб имеют неориентированный граф из n узлов и трех типов ребер:
Оптимизация кода с помощью тернарного оператора Python
Оптимизация кода с помощью тернарного оператора Python
И последнее, что мы хотели бы показать вам, прежде чем двигаться дальше, это
Советы по эффективной веб-разработке с помощью Python
Советы по эффективной веб-разработке с помощью Python
Как веб-разработчик, Python может стать мощным инструментом для создания эффективных и масштабируемых веб-приложений.
3
0
55
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

В первом сценарии генератор собирает мусор после next(), наконец срабатывает и вызывает исключение. Во втором генератор удерживается x, предотвращая это и позволяя нормально работать.

Редактировать:

Исключение GeneratorExit выдается, когда генератор закрывается, после чего будет выполнен блокfinally.

Также из документов:

Если генератор не возобновится до его завершения (по достижении нулевого счетчика ссылок или при сборке мусора), будет вызван метод close() генератора-итератора, позволяющий выполнить любые ожидающие предложенияfinally.

Я думаю, что это может быть правильно. Во втором примере, если вы следуете за next(x) с x = None, вы увидите exception и second.

Tom Karzes 05.07.2024 11:20

Почему же это вызывает исключение?

Joan Ràfols 05.07.2024 13:08

@JoanRàfols Я отредактировал свой ответ.

Coding Enthusiast 05.07.2024 14:03

Также см. этот ответ, в котором описана та же проблема и объясняется, почему возникает вопрос GeneratorExit.

Tom Karzes 05.07.2024 14:08

Другие вопросы по теме