В ES-2015 предусмотрены новые способы объявления переменных: через let
и const
вместо var
.
Например:
let a = 5;
let
У объявлений переменной через let
есть три основных отличия от var
:
-
Область видимости переменной
let
– блок{...}
.Как мы помним, переменная, объявленная через
var
, видна везде в функции.Переменная, объявленная через
let
, видна только в рамках блока{...}
, в котором объявлена.Это, в частности, влияет на объявления внутри
if
,while
илиfor
.Например, переменная через
var
:var apples = 5; if (true) { var apples = 10; alert(apples); // 10 (внутри блока) } alert(apples); // 10 (снаружи блока то же самое)
В примере выше
apples
– одна переменная на весь код, которая модифицируется вif
.То же самое с
let
будет работать по-другому:let apples = 5; // (*) if (true) { let apples = 10; alert(apples); // 10 (внутри блока) } alert(apples); // 5 (снаружи блока значение не изменилось)
Здесь, фактически, две независимые переменные
apples
, одна – глобальная, вторая – в блокеif
.Заметим, что если объявление
let apples
в первой строке(*)
удалить, то в последнемalert
будет ошибка: переменная не определена:if (true) { let apples = 10; alert(apples); // 10 (внутри блока) } alert(apples); // ошибка!
Это потому что переменная
let
всегда видна именно в том блоке, где объявлена, и не более. -
Переменная
let
видна только после объявления.Как мы помним, переменные
var
существуют и до объявления. Они равныundefined
:alert(a); // undefined var a = 5;
С переменными
let
всё проще. До объявления их вообще нет.Такой доступ приведёт к ошибке:
alert(a); // ошибка, нет такой переменной let a = 5;
Заметим также, что переменные
let
нельзя повторно объявлять. То есть, такой код выведет ошибку:let x; let x; // ошибка: переменная x уже объявлена
Это – хоть и выглядит ограничением по сравнению с
var
, но на самом деле проблем не создаёт. Например, два таких цикла совсем не конфликтуют:// каждый цикл имеет свою переменную i for(let i = 0; i<10; i++) { /* … */ } for(let i = 0; i<10; i++) { /* … */ } alert( i ); // ошибка: глобальной i нет
При объявлении внутри цикла переменная
i
будет видна только в блоке цикла. Она не видна снаружи, поэтому будет ошибка в последнемalert
. -
При использовании в цикле, для каждой итерации создаётся своя переменная.
Переменная
var
– одна на все итерации цикла и видна даже после цикла:for(var i=0; i<10; i++) { /* … */ } alert(i); // 10
С переменной
let
– всё по-другому.Каждому повторению цикла соответствует своя независимая переменная
let
. Если внутри цикла есть вложенные объявления функций, то в замыкании каждой будет та переменная, которая была при соответствующей итерации.Это позволяет легко решить классическую проблему с замыканиями, описанную в задаче Армия функций.
function makeArmy() { let shooters = []; for (let i = 0; i < 10; i++) { shooters.push(function() { alert( i ); // выводит свой номер }); } return shooters; } var army = makeArmy(); army[0](); // 0 army[5](); // 5
Если бы объявление было
var i
, то была бы одна переменнаяi
на всю функцию, и вызовы в последних строках выводили бы10
(подробнее – см. задачу Армия функций).А выше объявление
let i
создаёт для каждого повторения блока в цикле свою переменную, которую функция и получает из замыкания в последних строках.
const
Объявление const
задаёт константу, то есть переменную, которую нельзя менять:
const apple = 5;
apple = 10; // ошибка
В остальном объявление const
полностью аналогично let
.
Заметим, что если в константу присвоен объект, то от изменения защищена сама константа, но не свойства внутри неё:
const user = {
name: "Вася"
};
user.name = "Петя"; // допустимо
user = 5; // нельзя, будет ошибка
То же самое верно, если константе присвоен массив или другое объектное значение.
Константы, которые жёстко заданы всегда, во время всей программы, обычно пишутся в верхнем регистре. Например: const ORANGE = "#ffa500"
.
Большинство переменных – константы в другом смысле: они не меняются после присвоения. Но при разных запусках функции это значение может быть разным. Для таких переменных можно использовать const
и обычные строчные буквы в имени.
Итого
Переменные let
:
- Видны только после объявления и только в текущем блоке.
- Нельзя переобъявлять (в том же блоке).
- При объявлении переменной в цикле
for(let …)
– она видна только в этом цикле. Причём каждой итерации соответствует своя переменнаяlet
.
Переменная const
– это константа, в остальном – как let
.