Вы можете рассматривать функции как объекты и присваивать им «переменные-члены». Здесь мы используем внутреннюю функцию в fact
, но вместо того, чтобы просто объявить ее как локальную переменную (var loop = ...
), мы помещаем ее определение вне функции, используя синтаксис объекта (fact.loop = ...
). Это позволяет нам «экспортировать» внутреннюю функцию loop
из fact
, чтобы ее можно было повторно использовать функцией doubleFact
.
.
var fact = function(n) {
return fact.loop(n, 1);
};
fact.loop = function(n, acc) {
if (n < 1) {
return acc;
} else {
return fact.loop(n-1, acc * n);
}
};
var doubleFact = function(x) {
return fact.loop(x * 2, 1);
};
console.log(fact(5)); // 120
console.log(doubleFact(5)); // 3628800
Эту же идею можно использовать для поддержания состояния.
var countCalled = function() {
console.log("I've been called " + (++countCalled.callCount) + " times.");
};
countCalled.callCount = 0;
countCalled(); // I've been called 1 times.
countCalled(); // I've been called 2 times.
countCalled(); // I've been called 3 times.
Если вы хотите иметь возможность создавать несколько экземпляров, каждый со своим собственным состоянием, попробуйте следующее:
var CallCounter = function(name) {
var f = function() {
console.log(name + " has been called " + (++f.callCount) + " times.");
};
f.callCount = 0;
return f;
};
var foo = CallCounter("foo");
var bar = CallCounter("bar");
foo();
foo();
bar();
foo();
bar();
bar();
bar();
console.log(foo.callCount);
console.log(bar.callCount);
Выходы:
foo has been called 1 times.
foo has been called 2 times.
bar has been called 1 times.
foo has been called 3 times.
bar has been called 2 times.
bar has been called 3 times.
bar has been called 4 times.
3
4