Что такое Temporal Dead Zone (TDZ)?
Temporal Dead Zone (TDZ) — промежуток от начала блока до строки объявления let/const, где переменная существует, но недоступна. Обращение к ней выбрасывает ReferenceError.
Что такое Temporal Dead Zone
Когда интерпретатор входит в блок (функцию, if, for и т.д.), переменные let и const регистрируются в лексическом окружении немедленно, но инициализируются только при достижении строки let x = .... Промежуток между началом блока и строкой инициализации называется Temporal Dead Zone. Любое обращение к переменной в этот период вызывает ReferenceError.
{
// TDZ для x начинается здесь
console.log(typeof x); // ReferenceError: Cannot access 'x' before initialization
let x = 10; // TDZ заканчивается, x = 10
console.log(x); // 10
}
Сравнение с var
var поднимается (hoisting) и инициализируется undefined при входе в функцию — TDZ нет:
function demo() {
console.log(a); // undefined — var поднят и инициализирован
console.log(b); // ReferenceError — let в TDZ
console.log(c); // ReferenceError — const в TDZ
var a = 1;
let b = 2;
const c = 3;
}
demo();
TDZ в классах
Тело класса использует let/const-семантику; обращение к this до super() в конструкторе — тоже проявление TDZ-подобного ограничения:
class Animal {
constructor(name) {
this.name = name;
}
}
class Dog extends Animal {
constructor(name, breed) {
// console.log(this.name); // ReferenceError: Must call super before accessing 'this'
super(name);
this.breed = breed;
}
}
TDZ в параметрах функций по умолчанию
Параметры со значениями по умолчанию тоже имеют TDZ — нельзя ссылаться на более поздний параметр:
// Ошибка: b ещё в TDZ при вычислении значения a
function bad(a = b, b = 1) {
return a + b;
}
// bad() → ReferenceError
// Корректно: a объявлен раньше b
function good(a = 1, b = a * 2) {
return a + b;
}
good(); // 3
TDZ и typeof
Для var и необъявленных переменных typeof возвращает "undefined". Для переменных в TDZ — бросает ReferenceError:
typeof undeclaredVar; // "undefined" — безопасно
typeof x; // ReferenceError, если x — let в TDZ
// Практический вывод: проверка typeof "variable" !== "undefined"
// как old-school guard не работает с let/const
Диагностика в DevTools
// Типичная ошибка: замыкание + let в петле
const handlers = [];
for (let i = 0; i < 3; i++) {
handlers.push(() => console.log(i)); // OK: let создаёт новую переменную на каждой итерации
}
handlers.forEach(fn => fn()); // 0, 1, 2 — правильно
// С var была бы проблема:
const handlers2 = [];
for (var j = 0; j < 3; j++) {
handlers2.push(() => console.log(j));
}
handlers2.forEach(fn => fn()); // 3, 3, 3 — одна переменная j
Подводные камни
- TDZ возникает не только в начале блока, но и при каждой итерации блочного цикла —
letвforсоздаёт новую переменную на каждой итерации. typeofбольше не является безопасным способом проверки существования переменной, если она объявлена черезlet/const.- Circular imports в ES-модулях могут создавать TDZ-ситуации: модуль A импортирует из B, а B импортирует из A; при первичной загрузке A-экспорты могут быть в TDZ.
- Сообщение об ошибке в V8 — «Cannot access ... before initialization» — не всегда интуитивно указывает на TDZ; разработчики путают с hoisting.
- Class-декларации тоже имеют TDZ — класс нельзя использовать до строки его объявления, в отличие от function-декларации.
- Минификаторы (Terser) иногда переупорядочивают объявления и могут случайно вводить TDZ в неожиданных местах при агрессивных оптимизациях.
Common mistakes
- Смешивать «Temporal Dead Zone» с похожим механизмом без критерия выбора.
- Игнорировать риск: неверно оценить границы применения темы «Temporal Dead Zone» и получить хрупкое решение.
- Показывать только синтаксис и не объяснять поведение в runtime или сборке.
What the interviewer is testing
- Объясняет интервал между входом в область видимости и инициализацией
letилиconst. - Показывает на примере, как работает: переменная уже известна lexical environment, но доступ к ней до выполнения объявления приводит к ReferenceError; это предотвращает использование неинициализированного значения.
- Называет production-нюанс и граничный случай для темы «Temporal Dead Zone».