Таня Шлюссер - Автостопом по Python
- Название:Автостопом по Python
- Автор:
- Жанр:
- Издательство:Питер
- Год:2017
- Город:СПб.
- ISBN:978-5-496-03023-6
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Таня Шлюссер - Автостопом по Python краткое содержание
и содержит наработки многочисленных профессионалов и энтузиастов, знающих, что такое Python, и чего вы от него хотите. Проверенные методы и новейшие приемы, собранные в этой книге, помогут вам стать профессиональным Python-программистом и во всеоружии встретить наступающую эпоху Python 3.
Автостопом по Python - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
>>> some_lock.acquire()
>>> try:
… ····# Создать Землю 1, запустить ее на десять миллионов лет…
… ····print(
… ········"Look at me: I design coastlines.\n"
… ········"I got an award for Norway."
… ····)
… finally:
… ····some_lock.release()
Модуль стандартной библиотеки contextlib ( https://docs.python.org/3/library/contextlib.html) предоставляет дополнительные инструменты, которые помогают преобразовать функции в менеджеры контекстов, навязать вызов метода close(), подавить исключения (в Python 3.4 и выше) и перенаправить стандартные потоки вывода и ошибок (в Python 3.4, 3.5 и выше). Рассмотрим пример использования функции contextlib.closing():
>>> from contextlib import closing
>>> with closing(open("outfile.txt", "w")) as output:
… ····output.write("Well, he's…he's, ah…probably pining for the fjords.")
…
56
Но поскольку методы __enter__() и __exit__() определены для объекта, который отвечает за ввод/вывод для файла [41] В этом случае метод __exit__() вызовет метод обертки для ввода/вывода close(), чтобы закрыть дескриптор файла. Во многих системах имеется максимальное количество открытых дескрипторов файлов, и хороший тон — освобождать их по завершении работы.
, мы можем использовать это выражение непосредственно, не закрывая файл самостоятельно:
>>> with open("outfile.txt", "w") as output:
····output.write(
········"PININ' for the FJORDS?!?!?!? "
········"What kind of talk is that? look, why did he fall "
········"flat on his back the moment I got 'im home?\n"
····)
…
123
Распространенные подводные камни
По большей части Python — чистый и надежный язык. Однако некоторые ситуации могут быть непонятны для новичков: какие-то из них созданы намеренно, но все равно могут удивить, другие можно считать особенностями языка. В целом все, что продемонстрировано в этом подразделе, относится к неоднозначному поведению, которое может показаться странным на первый взгляд, но впоследствии выглядит разумным (когда вы узнаете о причинах).
Наиболее частый сюрприз, с которым сталкиваются новые программисты Python, — это отношение Python к изменяемым аргументам по умолчанию в определениях функции.
Что вы написали:
def append_to(element, to=[]):
····to.append(element)
····return to
Чего вы ожидаете:
my_list = append_to(12)
print(my_list)
my_other_list = append_to(42)
print(my_other_list)
Новый список создается всякий раз, когда вызывается функция, если второй аргумент не предоставлен, поэтому результат работы функции выглядит так:
[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.
Особенно неудобно то, что вам может показаться, будто ошибка как-то связана с лямбда-выражениями ( https://docs.python.org/3/tutorial/controlflow.html#lambda-expressions). Функции, создаваемые с помощью лямбда-выражений, не отличаются от других. Фактически то же самое поведение проявляется и при использовании самого обычного def:
def create_multipliers():
····multipliers = []
····for i in range(5):
········def multiplier(x):
············return i * x
········multipliers.append(multiplier)
····return multipliers
Что вам нужно сделать вместо этого? Наиболее общее решение, возможно, станет «костылем» — временным вариантом устранения проблемы. Из-за уже упомянутого поведения Python, связанного с определением аргументов по умолчанию для функций (см. предыдущий пункт «Изменяемые аргументы функций»), вы можете создать замыкание, которое немедленно связывается со своими аргументами с помощью аргумента по умолчанию:
def create_multipliers():
····return [lambda x, i=i: i * x for i in range(5)]
Помимо этого вы можете использовать функцию functools.partial():
from functools import partial
from operator import mul
def create_multipliers():
····return [partial(mul, i) for i in range(5)]
Когда подводный камень вовсе не подводный камень.Иногда нужно, чтобы замыкания вели себя подобным образом. Позднее связывание может быть полезным во многих ситуациях (например, в проекте Diamond, см. пункт «Пример использования замыкания (когда подводный камень вовсе не подводный камень)» на с. 136). Наличие уникальных функций в циклах, к сожалению, может привести к сбоям.
Читать дальшеИнтервал:
Закладка: