На этот вопрос уже ответил aaronasterling
Однако, кому-то может быть интересно узнать, как переменные хранятся под капотом.
Перед переходом к фрагменту:
Закрытия - это функции, которые наследуют переменные из окружающей их среды.Когда вы передаете функцию обратного вызова в качестве аргумента другой функции, которая будет выполнять ввод / вывод, эта функция обратного вызова будет вызвана позже, и эта функция - почти волшебным образом - запомнит контекст, в котором она была объявлена, вместе со всеми доступными переменнымив этом контексте.
Если функция не использует свободные переменные, она не образует замыкание.
Если есть другой внутренний уровень, который использует свободные переменные - все предыдущие уровни сохраняют лексическую среду (пример в конце)
атрибуты функции func_closure
в python <3.X или <code>__closure__ в python> 3.X сохраняют свободные переменные.
EveryФункция в Python имеет атрибуты замыкания, но не сохраняет никакого содержимого, если нет свободных переменных.
пример: атрибутов замыкания, но внутри нет содержимого, так какнет свободной переменной.
>>> def foo():
... def fii():
... pass
... return fii
...
>>> f = foo()
>>> f.func_closure
>>> 'func_closure' in dir(f)
True
>>>
Примечание: БЕСПЛАТНАЯ ПЕРЕМЕННАЯ ДОЛЖЕН СОЗДАТЬ ЗАКРЫТИЕ.
Я объясню, используятот же фрагмент, что и выше:
>>> def make_printer(msg):
... def printer():
... print msg
... return printer
...
>>> printer = make_printer('Foo!')
>>> printer() #Output: Foo!
И все функции Python имеют атрибут замыкания, поэтому давайте рассмотрим переменные, связанные с функцией замыкания.
Вот атрибут func_closure
для функцииprinter
>>> 'func_closure' in dir(printer)
True
>>> printer.func_closure
(<cell at 0x108154c90: str object at 0x108151de0>,)
>>>
Атрибут closure
возвращает набор объектов ячеек, которыесодержит сведения о переменных, определенных в прилагаемой области видимости.
Первый элемент в func_closure, который может иметь значение None, или кортеж ячеек, которые содержат привязки для свободных переменных функции и доступны только для чтения.
>>> dir(printer.func_closure[0])
['__class__', '__cmp__', '__delattr__', '__doc__', '__format__', '__getattribute__',
'__hash__', '__init__', '__new__', '__reduce__', '__reduce_ex__', '__repr__',
'__setattr__', '__sizeof__', '__str__', '__subclasshook__', 'cell_contents']
>>>
Здесь в приведенном выше выводе вы можете увидеть cell_contents
, давайте посмотрим, что он хранит:
>>> printer.func_closure[0].cell_contents
'Foo!'
>>> type(printer.func_closure[0].cell_contents)
<type 'str'>
>>>
Итак, когда мы вызвали функцию printer()
, она получает доступ к значению, хранящемуся внутриcell_contents
.Вот как мы получили вывод: «Foo!»
Снова я объясню использование приведенного выше фрагмента с некоторыми изменениями:
>>> def make_printer(msg):
... def printer():
... pass
... return printer
...
>>> printer = make_printer('Foo!')
>>> printer.func_closure
>>>
В приведенном выше фрагменте я не печатаю msgвнутри функции принтера, поэтому он не создает свободной переменной.Поскольку свободной переменной нет, внутри замыкания не будет содержимого.Это именно то, что мы видим выше.
Теперь я объясню другой фрагмент кода, чтобы очистить все Free Variable
с Closure
:
>>> def outer(x):
... def intermediate(y):
... free = 'free'
... def inner(z):
... return '%s %s %s %s' % (x, y, free, z)
... return inner
... return intermediate
...
>>> outer('I')('am')('variable')
'I am free variable'
>>>
>>> inter = outer('I')
>>> inter.func_closure
(<cell at 0x10c989130: str object at 0x10c831b98>,)
>>> inter.func_closure[0].cell_contents
'I'
>>> inn = inter('am')
Итак, мы видим, что func_closure
свойство является кортежем закрытия клеток , мы можем явно ссылаться на них и их содержимое - у ячейки есть свойство "cell_contents"
>>> inn.func_closure
(<cell at 0x10c9807c0: str object at 0x10c9b0990>,
<cell at 0x10c980f68: str object at 0x10c9eaf30>,
<cell at 0x10c989130: str object at 0x10c831b98>)
>>> for i in inn.func_closure:
... print i.cell_contents
...
free
am
I
>>>
Здесь, когда мы вызвали inn
, оно будетссылаемся на все переменные без сохранения, чтобы получить I am free variable
>>> inn('variable')
'I am free variable'
>>>