Автостопом по Python | страница 47
>[12]
>[42]
Что происходит на самом деле:
>[12]
>[12, 42]
Новый список создается при определении функции, он же используется в момент каждого последующего вызова: аргументы по умолчанию в Python оцениваются при определении функции, а не при каждом ее вызове (как это происходит, например, в Ruby).
Это означает, что если вы используете изменяемый по умолчанию аргумент и измените его, то он изменится для всех последующих вызовов этой функции.
Что вам нужно сделать вместо этого? Создавайте новый объект при каждом вызове функции, используя аргумент по умолчанию, чтобы показать, что аргумент не был передан (в качестве такого значения подойдет None):
>def append_to(element, to=None):
>····if to is None:
>········to = []
>····to.append(element)
>····return to
Когда подводный камень вовсе не подводный камень. Иногда вы можете намеренно задействовать (то есть использовать в качестве нормального варианта поведения) этот подводный камень, чтобы сохранять состояние между вызовами функции. Зачастую это делается при написании функции кэширования (которая сохраняет результаты в памяти), например:
>def time_consuming_function(x, y, cache={}):
>····args = (x, y)
>····if args in cache:
>········return cache[args]
>····# В противном случае функция работает с аргументами в первый раз.
>····# Выполняем сложную операцию…
>····cache[args] = result
>····return result
Еще один распространенный источник путаницы — способ связывания переменных в замыканиях (или в окружающей глобальной области видимости).
Что вы написали:
>def create_multipliers():
>····return [lambda x: i * x for i in range(5)]
Чего вы ожидаете:
>for multiplier in create_multipliers():
>····print(multiplier(2), end="… ")
>print()
Список, содержащий пять функций, каждая из них имеет собственную замкнутую переменную i, которая умножается на их аргумент, что приводит к получению следующего результата:
>0… 2… 4… 6… 8…
Что происходит на самом деле:
>8… 8… 8… 8… 8…
Создаются пять функций, все они умножают х на 4. Почему? В Python замыкания имеют позднее связывание. Это говорит о том, что значения переменных, использованных в замыканиях, определяются в момент вызова внутренней функции.
В нашем примере, когда вызывается любая из возвращенных функций, значение переменной i определяется с помощью окружающей области видимости в момент вызова. К этому моменту цикл завершает свою работу и i получает итоговое значение 4.
Особенно неудобно то, что вам может показаться, будто ошибка как-то связана с лямбда-выражениями (