Область видимости и замыкания в JavaScript
- воскресенье, 14 июля 2024 г. в 00:00:03
Тема довольно объемная и я не претендую на полное eё раскрытие в этой статье. Если вы хотите разобраться подробней, то искренне рекомендую вам книгу: Кайл Симпсон “Область видимости и замыкания”. Я был и научен и вдохновлен этой книгой. Все ссылки на ресурсы и книгу смотрите в конце.
Область видимости в JS — это любая область в коде, которая содержит именованные сущности (переменные, классы, функции) и определяет их доступность из разных частей кода.
На следующем примере есть две области видимости -- глобальная и область внутри функции:
// Глобальная область видимости
let number = 213;
function printNumber() {
// Область видимости функции printNumber
let color = "#f5f5f5"
console.log(number + number)
}
console.log(number)
console.log(printNumber())
// Переменна color не доступна в глобальной области видимости
// Она объявлена локально внутри функции printNumber
console.log(color)
Переменная number
объявлена в глобальной области видимости и поэтому доступна и внутри функции printNumber
. Сама функция printNumber
объявлена так же в глобальной области видимости и мы свободно ее вызываем.
Переменная color
объявлена внутри функции printNumber
и поэтому она не доступна в глобальной области видимости. Попытка обратиться к переменной color
вне функции printNumber
приведет к ошибке.
В итоге при запуске скрипта наблюдаем такую картину:
Пример выше демонстрирует, что мы получили: значение переменной number
, результат сложения внутри функции и undefined
, так как функция ничего не возвращает. При попытке обратиться к переменной color
консоль нам сообщает, что такая переменная не найдена. Такая ошибка универсальна. То есть если в нашем коде вообще не будет переменной color, содержание ошибки не изменится. В нашем случая можно сказать, что переменная не найдена в глобальной области видимости, но вообще в коде она присутствует.
Но почему тогда мы можем обратиться к переменной number
находясь в другой области видимости, но не можем таким же образом обратиться к переменной color
? Для начала изменим немного имеющийся код и добавим в него цвета.
// Глобальная область видимости
let number = 213;
function printNumber(operand) {
// Область видимости функции printNumber
let color = "#f5f5f5"
console.log(number + operand)
function printColor() {
// Область видимости функции printColor
console.log(color)
}
printColor()
}
console.log(number)
console.log(printNumber(312))
// Переменна color не доступна в глобальной области видимости
console.log(color)
// Функция printColor не доступна в глобальной области видимости
console.log(printColor())
Здесь я добавил параметр для функции printNumber
, использовал значение параметра при сложении и добавил функцию printColor
внутри функции printNumber
. Каждый цвет на картинке обозначает свою область видимости. Отсюда можно донести простую мысль: каждая вложенная область видимости имеет доступ ко всем внешним областям видимости. В свою очередь каждая внешняя область видимости не имеет доступа к вложенным областям. Получается что:
Оранжевая имеет доступ к розовой и зеленой
Розовая имеет доступ к зеленой
Зеленая (глобальная) не имеет внешней области видимости
Ели вы обратили внимание, область видимости функции начинается с момента объявления параметра. То есть когда мы передаём аргумент в функцию, в нашем случае printNumber(312)
, то этот аргумент присваивается параметру operand
и становится доступен только внутри функции.
Для глубокого понимания темы нужно изучить ряд концепций, которых здесь не будет, но которые есть в выше упомянутой книге Кайла Симпсона. А мы как и ранее пройдемся по поверхности.
Для начала скромный пример:
function main() {
let say = "I'am secondary function"
function secondary() {
console.log(say)
}
return secondary
}
const hi = main()
hi()
Ничего особенного не происходит. Мы возвращаем функцию secondary
из функции main
. Функция secondary
использует переменную say
, которая была объявлена в области видимости функции main
. В конце мы сохраняем результат вызова функции main
в переменную hi
. Так как результат вызова функции main
— это функция, то переменная hi
тоже становится функцией. Вызываем функцию hi
и видим в консоли значение переменной say
.
Функция hi
является всего лишь ссылкой на функцию secondary
. Вся красота в том, что до того как произойдет вызов hi()
, функция main
уже завершит выполнение. Если не вдаваться в подробности, а читать просто что написано, то следуя примитивной логике можно сказать, что вызов функции hi
должен привести к ошибке, так как функция main
уже завершила свою работу и переменной say
больше не существует. Вот тут мы и попадаем в замыкание. Учитывая пример можно сформулировать определение.
Замыкание — это способность функции запоминать своё окружение и взаимодействовать с ним в процессе выполнения кода, даже если эта функция вызывается вне того окружения, в котором была объявлена.
В нашем случае функция secondary
замыкается на окружении функции main
, так как использует переменную say
, которая объявлена во внешнем окружении (функции main
) относительно функции secondary
.
Даже если мы не будем возвращать функцию secondary
, а сразу вызовем ее внутри main
, а потом вызовем и саму main
, то это всё равно будет замыканием. Нам не обязательно возвращать замыкающуюся функцию.
function main() {
let say = "I'am secondary function"
function secondary() {
console.log(say)
}
secondary()
}
main()
Замыкания — это довольно частый приём в программирование. Например для меня, джуна самоучки, открытием было, что когда в React я прокидываю пропсы и использую их в компоненте — это замыкание.
Приведу несколько примеров, которые я реши привязать к простым игровым механикам.
Инкапсуляция
Представим что у нас есть игровой инвентарь. Нам нужно, что бы доступ к предметам этого инвентаря бы определен с помощью специальных функций и что бы ни каким другим способом нельзя было взаимодействовать с инвентарём. Для этого можно написать следующую функцию:
function inventory() {
let items = []
return {
pishItem(item) {
if (items.length === 10) {
console.log('The number of items in the inventory has reached its maximum')
return
}
items.push(item)
},
removeItem(item) {
if (items.length === 0 || !items.includes(item)) {
console.log('This item is not in the inventory')
return
}
items.splice(items.indexOf(item), 1);
},
getItems() {
console.log(items)
}
}
}
const actionsInventory = inventory()
actionsInventory.pishItem('book')
actionsInventory.pishItem('sword')
actionsInventory.pishItem('apple')
actionsInventory.pishItem('armor')
actionsInventory.removeItem('sword')
actionsInventory.getItems()
При первом чтении может выглядеть трудно, но по идее я модифицировал популярный пример счетчика, который инкапсулирует переменную count
:
function counter() {
let count = 0
return function choice(action) {
if (action === '+') {
count++
} else {
count--
}
console.log(count)
}
}
const changeCounter = counter()
changeCounter('+') // 1
changeCounter('-') // 0
changeCounter('+') // 1
В основном примере мы возвращаем объект с набором методов вместо одной функции. Каждая функция замыкается на массиве items
и совершает по одному действию. В итоге мы помещаем в инвентарь 4 предмета, затем один предмет удаляем и выводим весь список оставшихся предметов. Таким образом мы инкапсулировали массив items, при этом создали ограниченный набор действий для пользователя.
Callback
Функция callback — это функция, переданная в другую функцию в качестве аргумента, которая затем вызывается внутри внешней функции для выполнения какого-то действия.
Представим что в нашей неизвестной игре есть три вида сундуков: золотой, серебряный, бронзовый. В зависимости от типа сундука у игрока либо повышается, либо уменьшается шанс получить ресурсы.
function openChest(chestType, callback) {
console.log(`Открытие ${chestType} сундука...`);
setTimeout(() => {
let treasure;
if (chestType === "золотой") {
treasure = Math.random() > 0.3 ? "Ресурсы" : "ничего"; // 70% шанс найти ресурсы
} else if (chestType === "серебряный") {
treasure = Math.random() > 0.5 ? "Ресурсы" : "ничего"; // 50% шанс найти ресурсы
} else {
treasure = Math.random() > 0.7 ? "Ресурсы" : "ничего"; // 30% шанс найти ресурсы
}
if (treasure !== "ничего") {
callback(null, `Вы нашли ${treasure}!`);
} else {
callback("Сундук пустой");
}
}, 1000);
}
function startGame() {
openChest("золотой", (empty, result) => {
if (empty) {
console.log(empty);
} else {
console.log(result);
}
});
}
startGame();
Давайте начнем с момента вызова функции startGame
. Внутри функции startGame
вызывается функции openChest
, аргументами которой становятся тип сундука и анонимная функция. Эта анонимная функция представляет callback функцию, которая замыкается на переменной treasure
. Внутри функции openChest
мы имитируем асинхронный вызов. Вызов происходит в момент условной активации сундука игроком. В нашем коде setTimeot
отвечает за расчет награды и вот когда расчет заканчивается, срабатывает наш callback
, который сообщает о результате расчета.
Для понимая этого кода может понадобится чуть больше времени, чем для предыдущего примера и это нормально.
Мемоизация
Для начала давайте выясним что это такое в контексте разработки и рассмотрим базовый пример.
Мемоизация - это процесс проверки и кэширования вычисленных ранее значений, для предотвращения повторных вычислений
Классический пример:
function memoize(fn) {
const cache = new Map();
return function(...args) {
const key = JSON.stringify(args);
if (cache.has(key)) {
console.log('Кэшированный результат:', cache.get(key))
return cache.get(key);
}
const result = fn(...args);
cache.set(key, result);
return result;
};
}
Давайте разберем пример построчно, прежде чем рассмотреть версию, применимую на практике.
2 >> Создаём коллекцию cash
5 >> Сохраняем массив переданных аргументов в переменной key
6 >> Проверяем, вызывалась ли ранее функция с такими же аргументами
7, 8 >> Если вызывалась, то возвращаем кэшированный результат
9 >> Если расчет с такими аргументами еще не производился, то прокидываем аргументы дальше в функцию, выполняющую расчеты.
10 >> Кэшируем массив аргументов и результат расчетов
Теперь давайте посмотрим как функция мемоизации работает в боевых условиях. Представим что у нас есть прямоугольное игровое поле, поделенное на равные квадраты. На поле есть персонаж, а вокруг на некоторых участках поля появляются монеты. Для построения маршрута к монетам мы будем использовать условный алгоритм поиска пути. Наш алгоритм будет пустой функцией, которая возвращает массив состоящий из массива стартовых координат и массива координат выбранной монеты.
// Функция поиска пути
function findPath(start, goal, grid) {
// Этот код будет включать логику для нахождения пути от start до goal.
// Для упрощения примера оставим функциию пустой
// ...
// Возвращаем путь (список координат от start до goal)
return [start, goal];
}
// Функция мемоизации
function memoize(fn) {
const cache = new Map();
return function(...args) {
const key = JSON.stringify(args);
if (cache.has(key)) {
console.log('Кэшированный результат:', cache.get(key))
return cache.get(key);
}
const result = fn(...args);
cache.set(key, result);
return result;
};
}
const memoizedFindPath = memoize(findPath);
const start = [0, 0];
const goal = [5, 5];
const grid = [
// Представление игровой сетки (например, двумерный массив)
];
const path1 = memoizedFindPath(start, goal, grid);
console.log(path1); // "Путь от [0, 0] до [5, 5]"
// Повторный вызов findPath с теми же параметрами - результат будет взят из кэша
const path2 = memoizedFindPath(start, goal, grid);
console.log(path2); // "Путь от [0, 0] до [5, 5]"
// Вызов findPath с другими параметрами - результат будет вычислен и кэширован
const newStart = [1, 1];
const newGoal = [6, 6];
const path3 = memoizedFindPath(newStart, newGoal, grid);
console.log(path3); // "Путь от [1, 1] до [6, 6]"
Здесь может случиться недопонимание со строки 29. Происходит следующее: в переменную memoizedFindPath
мы сохраняем результат выполнения функции memoize
. Результат функции memoize
можно продемонстрировать следующим образом:
// ...
const memoizedFindPath = function(...args) {
const key = JSON.stringify(args)
if (cache.has(key)) {
console.log('Кэшированный результат:', cache.get(key))
return cache.get(key);
}
const result = fn(...args);
cache.set(key, result);
return result;
};
Разница лишь в том, что в изначальном примере, функция, которую возращает memoize
и которая сохраняется в memoizedFindPath
замыкается на переменной cache
и на переданной в качестве аргумента функции fn
. Функция, которую принимает memoize
(findPath), является вычислительной функцией и результат ее вычислений сохраняется в коллекцию cache
. Сама функция fn
замыкается на аргументах, которые будут переданы в созданную нами функцию memoizedFindPath
.
То есть:
В memoize
мы передаём аргумент в виде функции для вычислений и возвращаем функцию, которая проверяет входящие значения в виде аргументов значения и предотвращает повторные вычисления. Грубо говоря, memoize
является неким сервером, для постоянного доступа к cache
.
Теперь у нас есть функция memoizedFindPath
, которая проверяет проверяет входящие аргументы и исходя из результатов проверки следует одному из вариантов:
a. Если на основе переданных аргументов уже был получен результат, то memoizedFindPath
вернет этот результат без повторных вычеслений.
b. Если на основе переданных аргументов вычисления не производились, то memoizedFindPath
вызовет вычислительную функцию, сохранит результат вычислений в переменную result
, сохранит result
в коллекцию cache
в качестве значения и вернет переменную result
.
Оставлю ссылку на книгу Кайл Симпсон "Область видимости и замыкания"
Также мне нравится статья на доке
Замыкания и области видимости - довольна большая и не понятная для начинающих тема. Вам могут сказать, что замыкания сейчас используют редко и что вообще это знание, которое нужно только для того, что бы пройти собеседование. В корне с этим не согласен. Понимание того, как работает замыкание, погружает разработчика в глубинные основы языка, что позволяет решать специализированные задачи узкого профиля.
Специалистов, которые умеют писать базовый (поверхностный) код, очень много. А людей, которые умеют решать сложные задачи - значительно меньше.
Благодарю за прочтение, призываю к конструктивной критике и обсуждению в комментах.
Всем мир!