一、核心概念铺垫
首先明确两个基础概念,能帮你更好理解区别:
- 函数作用域:变量只在声明它的函数内部可访问(
var 是函数作用域)。
- 块级作用域:变量只在声明它的
{} 内部可访问(let/const 是块级作用域,{} 包括 if/for/while/普通代码块)。
- 变量提升:JS 引擎在执行代码前,会把变量声明'提升'到当前作用域顶部(但赋值不会提升)。
二、逐个拆解与对比
1. var(ES5 语法)
var 是 ES5 中声明变量的方式,特性如下:
- 作用域:函数作用域(无块级作用域)。
- 变量提升:声明会被提升,且提升后默认值为
undefined。
- 可重复声明:同一作用域内可重复声明同一个变量,后声明的会覆盖前一个。
- 可修改:声明后可随意修改值。
示例代码:
if (true) {
var a = 10;
}
console.log(a);
console.log(b);
var b = 20;
var c = 30;
var c = 40;
console.log(c);
var d = 50;
d = 60;
console.log(d);
2. let(ES6 语法)
let 是 ES6 为解决 var 的缺陷新增的声明方式,特性如下:
- 作用域:块级作用域(被
{} 限制)。
- 变量提升:存在'暂时性死区(TDZ)'——声明也会提升,但提升后不会赋值
undefined,在声明前访问会报错。
- 不可重复声明:同一作用域内不能重复声明同一个变量(包括不能和
var/const 重复)。
- 可修改:声明后可修改值。
示例代码:
if (true) {
let a = 10;
}
console.log(a);
console.log(b);
let b = 20;
let c = 30;
let d = 50;
d = 60;
console.log(d);
3. const(ES6 语法)
const 用于声明常量,特性和 let 基本一致,核心区别是不可修改:
- 作用域:块级作用域(和
let 完全一样)。
- 变量提升:同样有暂时性死区(和
let 完全一样)。
- 不可重复声明:和
let 完全一样。
- 不可修改:声明时必须赋值,且赋值后不能修改引用(注意:如果是对象/数组,内部属性/元素可修改)。
示例代码:
if (true) {
const a = 10;
}
console.log(a);
console.log(b);
const b = 20;
const c = 30;
const e = 50;
const obj = { name: "张三" };
obj.name = "李四";
console.log(obj);
const arr = [1, 2];
arr.push(3);
console.log(arr);
三、核心区别对比表
| 特性 | var | let | const |
|---|
| 作用域 | 函数作用域 | 块级作用域 | 块级作用域 |
| 变量提升 | 有(提升为 undefined) | 有(暂时性死区) | 有(暂时性死区) |
| 重复声明 | 允许 | 不允许 | 不允许 |
| 声明后赋值 | 允许(默认 undefined) | 允许(默认未初始化) | 不允许(必须立即赋值) |
| 修改值 | 允许 | 允许 | 引用不可改(对象/数组内部可改) |
四、实际开发中的使用建议
- 优先用
const:声明不需要修改的变量(如常量、固定引用的对象/数组),能减少代码意外修改的风险。
- 其次用
let:声明需要修改的变量(如循环变量、状态变量)。
- 完全不用
var:var 的函数作用域和变量提升容易导致 bug(比如循环变量泄露、重复声明覆盖),ES6 后已被 let/const 替代。
五、let 和 const 取代 var 的场景详解
let 和 const 是 ES6 新增的变量声明方式,核心是解决 var 的作用域和提升问题,它们的使用场景可以用一句话概括:优先用 const,需要修改时用 let。下面结合具体场景和代码示例,帮你彻底分清什么时候用哪个。
1. const 的使用场景(优先选择)
const 声明的是引用不可变的变量(声明时必须赋值,且不能重新赋值),适合以下所有'不需要修改值/引用'的场景:
1.1 声明常量(固定不变的值)
比如数学常量、配置项、固定字符串等,用 const 能明确表示'这个值不会变',减少代码意外修改的风险。
const PI = 3.1415926;
const RADIUS = 5;
const AREA = PI * RADIUS * RADIUS;
const BASE_URL = 'https://api.example.com';
const TOKEN = 'abc123456';
const TITLE = '个人信息';
1.2 声明不需要重新赋值的对象/数组
注意:const 只是限制'引用不可变',对象/数组的内部属性/元素可以修改,这种场景非常适合用 const(因为你只是操作内部,不会重新赋值整个对象/数组)。
const userInfo = { name: '张三', class: '计算机专业', hobby: ['唱歌', '跳舞'] };
userInfo.name = '小雨';
userInfo.hobby.push('爬山');
const list = ['苹果', '香蕉'];
list.push('橙子');
1.3 声明 DOM 元素(创建后不会重新赋值)
创建 DOM 元素后,通常只会修改它的文本/样式,不会重新赋值成另一个元素,所以用 const 最合适。
const h2 = document.createElement('h2');
h2.textContent = '个人信息';
document.body.appendChild(h2);
const btn = document.createElement('button');
btn.textContent = '点击';
document.body.appendChild(btn);
2. let 的使用场景(仅当需要修改时)
let 声明的是可重新赋值的变量,只有当你需要'修改变量的值/引用'时,才用 let,常见场景如下:
2.1 循环变量(值会不断变化)
for/while 循环中的计数器(比如 i),每次循环都会修改值,必须用 let(不能用 const,因为 i++ 是重新赋值)。
const info = ['班级', '学号', '姓名', '性别', '爱好'];
for (let i = 0; i < info.length; i++) {
const p = document.createElement('p');
p.textContent = info[i];
document.body.appendChild(p);
}
let count = 0;
while (count < 3) {
console.log('计数:', count);
count++;
}
2.2 状态变量(值会动态变化)
比如开关状态、计数器、表单输入值等,这些值会随着操作(点击、输入)变化,需要用 let。
let isShow = false;
const btn = document.createElement('button');
btn.textContent = '显示/隐藏';
btn.onclick = function() {
isShow = !isShow;
console.log('状态:', isShow);
};
document.body.appendChild(btn);
let num = 0;
const countBtn = document.createElement('button');
countBtn.textContent = '计数';
countBtn.onclick = function() {
num++;
countBtn.textContent = `计数:${num}`;
};
document.body.appendChild(countBtn);
2.3 临时变量(需要重新赋值)
比如分步计算的值、条件判断后需要修改的变量,这些场景需要多次赋值。
let total = 0;
total += 10;
total += 20;
total *= 0.8;
console.log('总价:', total);
let message;
const isLogin = false;
if (isLogin) {
message = '欢迎登录';
} else {
message = '请先登录';
}
3. 避坑提醒(新手常错)
- ❌ 错误:用
const 声明需要修改的变量
const i = 0;
i++;
- ❌ 错误:没必要用
let 声明不修改的变量
let TITLE = '个人信息';
总结
- 优先用
const:所有不需要重新赋值的场景(常量、对象/数组、DOM 元素、固定字符串等),用 const 能让代码更安全、语义更清晰。
- 仅用
let:只有当变量需要重新赋值时(循环变量、状态变量、临时变量等),才用 let。
- 核心原则:能不用
let 就不用,用 const 约束'不变',用 let 表示'可变',彻底抛弃 var。