Найти - Пользователи
Полная версия: Окружение в замыкании
Начало » Python для новичков » Окружение в замыкании
1
PooH
Может кто объяснить? Следующий код:
class CategoryWidget(forms.CheckboxSelectMultiple):

def render(self, name, value, attrs=None, choices=()):
final_attrs = self.build_attrs(attrs, name=name)
output = [u'<ul>']
print 1, locals()

def render_list(choices):
print 2, locals()
for i, (option_value, option_label, children) in enumerate(choices):
final_attrs = dict(final_attrs, id='%s_%s' % (attrs['id'], i))
label_for = u' for="%s"' % final_attrs['id']
cb = forms.CheckboxInput(final_attrs, check_test=lambda value: value in str_values)
option_value = force_unicode(option_value)
rendered_cb = cb.render(name, option_value)
option_label = conditional_escape(force_unicode(option_label))
output.append(u'<li><label%s>%s %s</label></li>' % (label_for, rendered_cb, option_label))

render_list(chain(self.choices, choices))
output.append(u'</ul>')
return mark_safe(u'\n'.join(output))
выводит:

1 {'name': ‘sectors’, ‘self’: <zondir.base.forms.CategoryWidget object at 0x91512cc>, ‘value’: None, ‘choices’: (), ‘attrs’: {'id': u'id_sectors'}, ‘output’: , ‘final_attrs’: {'name': ‘sectors’, ‘id’: u'id_sectors'}}
2 {'output': , ‘attrs’: {'id': u'id_sectors'}, ‘name’: ‘sectors’, ‘choices’: <itertools.chain object at 0x91d9ccc>}
Почему внутри вложенной функции доступно output, но недоступно final_attrs?
shupg
Упрощенный код
def g():
a = [2]
print locals()

def f():
print locals()
a.append(a)
print locals()
f()

g()
->{'a': }
{'a': }
{'a': [2, ]}

def g():
a = [2]
print locals()

def f():
print locals()
a = 3
print locals()

f()

g()
->{'a': }
{}
{'a': 3}

Разница удивила, но почему оно так работает!
Zubchick
shupg
Упрощенный код
в первом случае ‘а’ не находится в локальном окружении и ищется выше. Во втором случае ‘а’ не находится в локальном окружении, после чего создается в нем и имя ‘a’ связывается с 3. В первом случае список ‘а’ не создается внутри функции, потому что чтобы обратиться к методу объекта (список) нужно его сначала создать.
shupg
Zubchick
Спасибо за ответ.
Уточню вопрос

def g():
a = [2]
def f():
print a
f()
g()
->

def g():
a = [2]
def f():
del a
f()
g()
->UnboundLocalError: local variable ‘a’ referenced before assignment

Почему в одном случае
Zubchick
не находится в локальном окружении и ищется выше
, а в другом нет?
Zubchick
Наверно что бы функции расположенные внутри по возможности не влияли на внешние, так как это во-первых ужасно читается, во вторых ведет к ошибкам. Чтобы что-то поменять, передавайте это параметром и возвращайте return'ом, это правильный путь. Ну или в случае с замыканиями, параметром можно не передавать.
PooH
И все таки, почему final_attrs отсутствует в замыкании? Я специально весь кусок кода привел - может я просто не вижу явного косяка?
o7412369815963
PooH
И все таки, почему final_attrs отсутствует в замыкании? Я специально весь кусок кода привел - может я просто не вижу явного косяка?
видимо в этом:
> final_attrs = dict(final_attrs, id='%s_%s' % (attrs, i))
Ed
PooH
И все таки, почему final_attrs отсутствует в замыкании? Я специально весь кусок кода привел - может я просто не вижу явного косяка?
Разница между output и final_attrs в вашем коде только в том, что final_attrs присваивается значение, а output меняется. То есть final_attrs воспринимается как новая переменная в контексте замыкания, а output - как уже существующая из внешнего контекста.

Вот вам пример:
def foo():
var1 = []
var2 = []
print 1, locals()

def closure1():
print 2, locals()
var1.append(1)
var2 = []

def closure2():
print 3, locals()
var1 = []
var2.append(1)

closure1()
closure2()

foo()
и его результат:
1 {'var1': , ‘var2’: }
2 {'var1': }
3 {'var2': }
PooH
Спасибо всем. Понял. А питон-то все таки опасная змеюка, может цапнуть иногда ;)
Ferroman
В “Python Essential Reference, Fourth Edition” хорошо такие моменты высветлены.
This is a "lo-fi" version of our main content. To view the full version with more information, formatting and images, please click here.
Powered by DjangoBB