var、let、const 的核心区别在于作用域、赋值约束和声明行为:var 仅函数作用域且变量提升;let 具块级作用域与暂时性死区;const 声明不可重赋值的绑定,但引用类型内部仍可变。

var、let 和 const 的核心区别不是“哪个更高级”,而是作用域规则、赋值约束和声明行为不同——选错一个,轻则逻辑出 bug,重则在 for 循环或条件块里拿到意料之外的值。
为什么 var 在 if 块里声明后,外面还能访问?
因为 var 只有函数作用域(或全局作用域),没有块级作用域。它会无视 {},直接提升到最近的函数顶部。
var a = 'outer';
if (true) {
var a = 'inner';
}
console.log(a); // "inner" —— 被覆盖了,且无报错
常见坑:
• 在循环中用 var i,所有闭包共享同一个 i,导致异步回调全输出最后一个值;
• 同名变量在多个 if 或 for 中重复 var,不会报错但容易覆盖;
• console.log(x) 放在 var x = 1 前面,不报错但输出 undefined(变量提升)。
为什么 let 声明后,提前访问就报 ReferenceError?
let 存在「暂时性死区」(TDZ):从块开头到声明语句执行前,该变量不可访问。
console.log(b); // Uncaught ReferenceError: Cannot access 'b' before initialization let b = 2;
实操建议:
• 所有局部变量,只要不需要函数作用域,优先用 let;
• 在 for (let i = 0; ...) 中,每次迭代都生成独立的 i,闭包安全;
• 不允许重复声明:let x = 1; let x = 2; → 报错;
• 允许后续赋值:let x = 1; x = 2; ✅ 正常。
const 真的是“不可变”吗?为什么对象属性还能改?
const 约束的是绑定关系(即变量名指向的内存地址不能变),不是值本身是否可变。
立即学习“Java免费学习笔记(深入)”;
const obj = { name: 'Ken' };
obj.name = 'KenNaNa'; // ✅ 合法:修改对象内容
obj = {}; // ❌ TypeError:不能重新赋值
const arr = [1, 2];
arr.push(3); // ✅ 合法:数组内容可变
arr = []; // ❌ 报错
关键点:
• const 必须在声明时初始化,否则语法错误;
• 基本类型(string/number/boolean)用 const 就真不可变;
• 引用类型(Object/Array/Function)用 const 只防重赋值,不防内部修改;
• 如果需要真正冻结对象,得配合 Object.freeze()。
实际项目中怎么选?一句话决策树
先问自己三个问题:
• 这个变量后续还会重新赋值吗?→ 是 → 用 let;否 → 看下一条;
• 它是在函数外/全局用,还是只在某个 if、for 或 {} 内使用?→ 后者 → 用 const(或 let);前者且需全局暴露 → 才考虑 var(但现代项目基本不用);
• 它是不是配置项、API 地址、DOM 节点引用等“只读标识符”?→ 是 → const 最稳妥。
最常被忽略的一点:var 在模块顶层(如 ES Module 文件顶部)声明,不会挂到 window 上;但在 script 标签全局作用域里,var a = 1 仍等于 window.a = 1 —— 这个隐式绑定在现代工程中极易引发命名冲突,能避则避。











