Javascript-форум (https://javascript.ru/forum/)
-   Общие вопросы Javascript (https://javascript.ru/forum/misc/)
-   -   Вопрос по замыканиям (https://javascript.ru/forum/misc/15644-vopros-po-zamykaniyam.html)

Goodfella 07.03.2011 21:55

Вопрос по замыканиям
 
Добрый вечер друзья, возник вопрос, скорей даже не вопрос, а полная неразбериха по теме замыканий (недавно в JS). Что бы без длинных предысторий, сразу покажу код (взял из Википедии):
function createCounter() {
    var numberOfCalls = 0;
    return function() {
        return ++numberOfCalls;
    }
}
var fn = createCounter();
fn(); //1
fn(); //2
fn(); //3

Я понимаю, что этот код делает, но абсолютно не понимаю каким образом.
Почему при вызове функции createCounter() не обнуляется переменная numberOfCalls?
Зачем нужно копировать ссылку на функцию в переменную fn, почему ее нельзя вызвать напрямую?
Сам по себе, этот код меня не волнует, но это самый простой пример замыкания, который я, к сожалению, не могу понять. Я бы был очень благодарен, если бы вы подробно объяснили мне ситуацию, как это все считывает интерпретатор.

Kolyaj 07.03.2011 21:59

Цитата:

Сообщение от Goodfella
Почему при вызове функции createCounter() не обнуляется переменная numberOfCalls?

Он обнуляется.

Цитата:

Сообщение от Goodfella
Зачем нужно копировать ссылку на функцию в переменную fn, почему ее нельзя вызвать напрямую?

Ссылка на функцию не копируется. Переменной fn присваивается результат работы функции createCounter().

Goodfella 07.03.2011 22:07

Цитата:

Сообщение от Kolyaj
Он обнуляется.

Меня ввел в недоумение метод alert(numberOfCalls), если его поставить перед
return ++numberOfCalls;

При каждом вызове функции, выводит увеличенный на единицу (по сравнению с предыдущим вызовом) результат, то есть - не обнуляется.

Kolyaj 07.03.2011 22:40

При вызове функции createCounter обнуляется. Она у вас только один раз вызывается.

Goodfella 07.03.2011 22:53

Цитата:

Сообщение от Kolyaj
При вызове функции createCounter обнуляется. Она у вас только один раз вызывается.

Да, спасибо, я чуть-чуть посидел - разобрался с этим, но все же, если мы записываем в переменную fn именно значение возвращаемое функцией createCounter(), что значит fn()? Я это воспринимаю как "вызвать значение", что есть абсурд.

Aetae 07.03.2011 22:54

Вот так будет работать как вы хотите, только это не имеет смысла))
function createCounter() {
    var numberOfCalls = 0;
    return function() {
        return ++numberOfCalls;
    }
}
var fn = createCounter;
fn()(); //1
fn()(); //1
fn()(); //1

Kolyaj 07.03.2011 22:58

Цитата:

Сообщение от Goodfella
Я это воспринимаю как "вызвать значение", что есть абсурд.

Почему абсурд, если значением является функция? А createCounter возвращает функцию.

Goodfella 07.03.2011 23:13

Цитата:

Сообщение от Kolyaj
Почему абсурд, если значением является функция? А createCounter возвращает функцию.

Ага! То есть, фактически, fn() вызывает функцию
function() {
    return ++numberOfCalls;
}

Спасибо большое, вы очень помогли!
Если бы еще понять вот этот синтаксис, fn()(), который все таки вызывает фукцию createCounter(), но я думаю, это уже не так проблематично.

Aetae 08.03.2011 01:09

var fn = createCounter;

Вот в чем дело, лол.)

А по поводу fn()() - ты это сколько угодно:
function џ() {
    return function() {
        return function() {
            return function() {
                return function() {
                    return function() {
                        return function() {
                            return function() {
                                alert('lol')
                            }
                        }
                    }
                }
            }
        }
    }
}
џ()()()()()()()()

хотя всё это фигнестрадание)

Goodfella 08.03.2011 02:29

Цитата:

Сообщение от Aetae
хотя всё это фигнестрадание)

Это ради понимания синтаксиса, не для решения конкретной задачи, но принцип уловил, спасибо.


Часовой пояс GMT +3, время: 12:01.