2016-03-30 2 views
0

Следующий код:Почему переменные в закрытии не забыты?

package main 

import "fmt" 

// fibonacci is a function that returns 
// a function that returns an int. 

func fibonacci() func() int { 
    first, second := 0, 1 

    return func() int { 
     // return next fibonacci number here. 
     first, second = second, first+second 
     return first 
    } 
} 

func main() { 
    f := fibonacci() 
    for i := 0; i < 10; i++ { 
     fmt.Println(f()) 
    } 
} 

возвращает 10 числа последовательности Фибоначчи. Что меня смущает, так это то, что это работы. Кажется, что значения first и second являются как-то сохранены в памяти, так как каждый раз, когда код выполняется, возвращается новое число фибоначчи в последовательности с предыдущим. Я думал, что функции потеряли свои запоминаемые переменные, когда они были выполнены. Что здесь происходит?

+2

Ну, вот и вся точка закрытия. Я могу порекомендовать прочитать [this] (http://stackoverflow.com/questions/111102/how-do-javascript-closures-work?rq=1) (это для Javascript, а не Go, но это не имеет никакого значения здесь). – tur

ответ

1

first и second являются переменными в fibonacci() FUNC, которые были «закрыты над» возвращенным func() int, который был возвращен из fibonacci().

Таким образом, они находятся в закрытии, связанном с f, поэтому f имеет доступ к этим переменным, если существует.

См. this Перейти на слайд (и те, что вокруг него) для объяснения причин закрытия Go.