| 
 Dmitry A. Soshnikov, Спасибо за подробные размышления, чем больше думаю, тем больше слоняюсь к мысли, что условия не нужны, в смысле "все функции замыкания и точка". ;) | 
| 
 Цитата: 
 Но, сегодня более глубоко посмотрел, как это дело обстоит в Python'е (да-да, опять Питон ;)). Так вот. Там так же работает механизм цепи скопов (т.е. из вложенных функций мы можем достучаться до всех вышестоящих переменных/объектов, включая глобальный скоп). Однако, начиная с версии Python'a 3.0, у функций есть интересное свойство - __closure__, которое: Цитата: 
 Это свойство, как было сказано, хранит кортеж (tuple, неизменяемый массив) свободных переменных функции. Т.е. это тот самый [[scope]] из JS, который записался в функцию при её создании, только здесь мы имеем к нему доступ явно. Но! Самое интересное здесь то, что глобальные переменные туда не входят! Здесь, если функция глобальная - у неё, считается, нет свободных лексических локальных переменных (и в этом случае, __closure__ == None): 
a = 10
# глобальная функция
def x():
    print(a)
x() # 10, работает, но:
x.__closure__ # None
А теперь с вложенными функциями: 
a = 10
def x():
    b = 20
    def y(): # первая вложенная
        c = 30
        def z(): # вторая вложенная
            print(a, b, c)
        return z # возврат из y
    return y # возврат из x
# проверка
dy = x() # вернувшаяся функция "y"
dz = dy() # вернувшаяся функция "z", можно было сразу - x()()
dz() # 10, 20, 30
# замыкание внешней функции "x"
x.__closure__ # None
# замыкание внутренней функции "y" (dy)
dy.__closure__ # (<cell at 0x0143AE70: int object at 0x1E1E5678>,) - кортеж из одного элемента
dy.__closure__[0] # <cell at 0x0143AE70: int object at 0x1E1E5678> - этот самый элемент, объект cell
dy.__closure__[0].cell_contents # а вот она наша 20!
dy.__closure__[1] # ошибка! нет элемента с индексом 1, только одна свободная переменная, глобальная "а" сюда не входит
# замыкание внутренней функции "z" (dz)
dz.__closure__ # два объекта, как и ожидалось - (<cell at 0x0143E970: int object at 0x1E1E5718>, <cell at 0x0143AE70: int object at 0x1E1E5678>)
dz.__closure__[0].cell_contents # 30
dz.__closure__[1].cell_contents # 20
dz.__closure__[2].cell_contents # ошибка, только две свободные переменные
Т.е. здесь, обособление лексических переменных от глобальных представлено явно. И именно лексические переменные попали в замыкание. P.S.:> Zeroglif, скачай, поставь интерпретатор Python'a, тоже, как и JS, интересно. P.S.[2]:> сейчас ещё и Ruby посмотрел - там тоже есть явное разделение на глобальные переменные (префикс - $: $a, $b и т.д.), и локальные - без префиксов. Так вот глобальные переменные могут использоваться в методах (def'ы), в то время как локальные для глобального контекста - нет: a = 10 def x puts a end x # ошибка, переменная "а" не определена $b = 10 def y puts $b end y # 10, все Ок В свою очередь замыкания в Ruby, которые создаются через block'и, lambda'ы и Proc'ы, запоминают эти лексические переменные: 
a = 10
x = lambda {
  puts a
}
x.call # 10
В общем, сколько идей, столько и реализаций ;) | 
| Часовой пояс GMT +3, время: 21:59. |