JavaScript 中 var、let、const 的核心区别与实战应用

JavaScript 中 var、let、const 的核心区别与实战应用

要理解 constvarlet 的区别,我们可以从 作用域、变量提升、可重复声明、可修改性 这几个核心维度展开,这些也是新手最容易混淆的点。

一、核心概念铺垫

首先明确两个基础概念,能帮你更好理解区别:

  • 函数作用域:变量只在声明它的函数内部可访问(var 是函数作用域)。
  • 块级作用域:变量只在声明它的 {} 内部可访问(let/const 是块级作用域,{} 包括 if/for/while/ 普通代码块)。
  • 变量提升:JS 引擎在执行代码前,会把变量声明 “提升” 到当前作用域顶部(但赋值不会提升)。

二、逐个拆解 + 对比

1. var(ES5 语法)

var 是 ES5 中声明变量的方式,特性如下:

  • 作用域:函数作用域(无块级作用域)。
  • 变量提升:声明会被提升,且提升后默认值为 undefined
  • 可重复声明:同一作用域内可重复声明同一个变量,后声明的会覆盖前一个。
  • 可修改:声明后可随意修改值。

示例代码

// 1. 函数作用域(无块级作用域) if (true) { var a = 10; } console.log(a); // 输出 10(a 跳出了 if 的 {},因为 var 无块级作用域) // 2. 变量提升 console.log(b); // 输出 undefined(声明提升了,赋值没提升) var b = 20; // 3. 可重复声明 var c = 30; var c = 40; console.log(c); // 输出 40(后声明的覆盖前一个) // 4. 可修改 var d = 50; d = 60; console.log(d); // 输出 60 
2. let(ES6 语法)

let 是 ES6 为解决 var 的缺陷新增的声明方式,特性如下:

  • 作用域:块级作用域(被 {} 限制)。
  • 变量提升:存在 “暂时性死区(TDZ)”—— 声明也会提升,但提升后不会赋值 undefined,在声明前访问会报错。
  • 不可重复声明:同一作用域内不能重复声明同一个变量(包括不能和 var/const 重复)。
  • 可修改:声明后可修改值。

示例代码

// 1. 块级作用域 if (true) { let a = 10; } console.log(a); // 报错:a is not defined(a 被 if 的 {} 限制,外部访问不到) // 2. 暂时性死区 console.log(b); // 报错:Cannot access 'b' before initialization let b = 20; // 3. 不可重复声明 let c = 30; // let c = 40; // 报错:Identifier 'c' has already been declared // var c = 40; // 同样报错 // 4. 可修改 let d = 50; d = 60; console.log(d); // 输出 60 
3. const(ES6 语法)

const 用于声明常量,特性和 let 基本一致,核心区别是不可修改

  • 作用域:块级作用域(和 let 完全一样)。
  • 变量提升:同样有暂时性死区(和 let 完全一样)。
  • 不可重复声明:和 let 完全一样。
  • 不可修改:声明时必须赋值,且赋值后不能修改引用(注意:如果是对象 / 数组,内部属性 / 元素可修改)。

示例代码

// 1. 块级作用域 if (true) { const a = 10; } console.log(a); // 报错:a is not defined // 2. 暂时性死区 console.log(b); // 报错:Cannot access 'b' before initialization const b = 20; // 3. 不可重复声明 const c = 30; // const c = 40; // 报错 // 4. 声明时必须赋值 // const d; // 报错:Missing initializer in const declaration // 5. 不可修改引用(重点) const e = 50; // e = 60; // 报错:Assignment to constant variable. // 注意:对象/数组的内部可修改(只是引用不能变) const obj = { name: "张三" }; obj.name = "李四"; // 正常执行,输出 { name: '李四' } console.log(obj); const arr = [1, 2]; arr.push(3); // 正常执行,输出 [1,2,3] console.log(arr); 

三、核心区别对比表

特性varletconst
作用域函数作用域块级作用域块级作用域
变量提升有(提升为 undefined)有(暂时性死区)有(暂时性死区)
重复声明允许不允许不允许
声明后赋值允许(默认 undefined)允许(默认未初始化)不允许(必须立即赋值)
修改值允许允许引用不可改(对象 / 数组内部可改)

四、实际开发中的使用建议

  1. 优先用 const:声明不需要修改的变量(如常量、固定引用的对象 / 数组),能减少代码意外修改的风险。
  2. 其次用 let:声明需要修改的变量(如循环变量、状态变量)。
  3. 完全不用 varvar 的函数作用域和变量提升容易导致 bug(比如循环变量泄露、重复声明覆盖),ES6 后已被 let/const 替代。

let 和 const  取代var

letconst 是 ES6 新增的变量声明方式,核心是解决 var 的作用域和提升问题,它们的使用场景可以用一句话概括:优先用 const,需要修改时用 let。下面结合具体场景和代码示例,帮你彻底分清什么时候用哪个。

一、const 的使用场景(优先选择)

const 声明的是引用不可变的变量(声明时必须赋值,且不能重新赋值),适合以下所有 “不需要修改值 / 引用” 的场景:

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 = '个人信息'; // 就像你之前写的标题文本 
2. 声明不需要重新赋值的对象 / 数组

注意:const 只是限制 “引用不可变”,对象 / 数组的内部属性 / 元素可以修改,这种场景非常适合用 const(因为你只是操作内部,不会重新赋值整个对象 / 数组)。

// 个人信息对象(只会改内部属性,不会重新赋值) const userInfo = { name: '韩雨龙', class: '24专本软件', hobby: ['唱歌', '跳舞'] }; // 允许修改内部属性(符合 const 规则) userInfo.name = '小雨'; userInfo.hobby.push('爬山'); // 列表数据(只会增删元素,不会重新赋值数组) const list = ['苹果', '香蕉']; list.push('橙子'); // 正常执行 
3. 声明 DOM 元素(创建后不会重新赋值)

就像你之前写的创建 h2 元素,创建后只会修改它的文本 / 样式,不会重新赋值成另一个元素,所以用 const 最合适。

// 创建 h2 元素(不会重新赋值,用 const) const h2 = document.createElement('h2'); h2.textContent = '个人信息'; // 只是修改属性,不是重新赋值 document.body.appendChild(h2); // 创建按钮元素(同理) const btn = document.createElement('button'); btn.textContent = '点击'; document.body.appendChild(btn); 

二、let 的使用场景(仅当需要修改时)

let 声明的是可重新赋值的变量,只有当你需要 “修改变量的值 / 引用” 时,才用 let,常见场景如下:

1. 循环变量(值会不断变化)

for/while 循环中的计数器(比如 i),每次循环都会修改值,必须用 let(不能用 const,因为 i++ 是重新赋值)。

// for 循环计数器(i 会从 0 变到 4,需要修改) const info = ['班级', '学号', '姓名', '性别', '爱好']; for (let i = 0; i < info.length; i++) { const p = document.createElement('p'); p.textContent = info[i]; document.body.appendChild(p); } // while 循环变量 let count = 0; while (count < 3) { console.log('计数:', count); count++; // 重新赋值,必须用 let } 
2. 状态变量(值会动态变化)

比如开关状态、计数器、表单输入值等,这些值会随着操作(点击、输入)变化,需要用 let

// 开关状态(点击按钮切换 true/false) let isShow = false; const btn = document.createElement('button'); btn.textContent = '显示/隐藏'; btn.onclick = function() { isShow = !isShow; // 重新赋值,用 let console.log('状态:', isShow); }; document.body.appendChild(btn); // 计数器(点击一次加 1) let num = 0; const countBtn = document.createElement('button'); countBtn.textContent = '计数'; countBtn.onclick = function() { num++; // 重新赋值,用 let countBtn.textContent = `计数:${num}`; }; document.body.appendChild(countBtn); 
3. 临时变量(需要重新赋值)

比如分步计算的值、条件判断后需要修改的变量,这些场景需要多次赋值。

// 分步计算总价 let total = 0; total += 10; // 加商品1价格 total += 20; // 加商品2价格 total *= 0.8; // 打8折(多次重新赋值,用 let) console.log('总价:', total); // 条件赋值(根据条件修改变量) let; const isLogin = false; if (isLogin) { message = '欢迎登录'; } else { message = '请先登录'; // 重新赋值,用 let } 

三、避坑提醒(新手常错)

  1. ❌ 错误:用 const 声明需要修改的变量
const i = 0; i++; // 报错:Assignment to constant variable.(const 不能重新赋值) 
  1. ❌ 错误:没必要用 let 声明不修改的变量
// 标题文本不会改,应该用 const 而不是 let let TITLE = '个人信息'; 

总结

  1. 优先用 const:所有不需要重新赋值的场景(常量、对象 / 数组、DOM 元素、固定字符串等),用 const 能让代码更安全、语义更清晰。
  2. 仅用 let:只有当变量需要重新赋值时(循环变量、状态变量、临时变量等),才用 let
  3. 核心原则:能不用 let 就不用,用 const 约束 “不变”,用 let 表示 “可变”,彻底抛弃 var

Read more

openclaw喂饭教程!在 Linux 环境下快速完成安装、初始化与 Web UI 配置

openclaw喂饭教程!在 Linux 环境下快速完成安装、初始化与 Web UI 配置

前言 OpenClaw 是一款开源的 AI Agent 工具,但对第一次接触的用户来说,完整跑通流程并不直观。本文以 Linux 环境为例,详细记录了 OpenClaw 的安装、初始化流程、模型选择、TUI 使用方式,以及 TUI 与 Web UI 认证不一致导致的常见问题与解决方法,帮助你最快速度把 OpenClaw 真正跑起来 环境准备 1)安装nodejs curl -fsSL https://deb.nodesource.com/setup_22.x | sudo -E bash - sudo apt install -y nodejs > node

前端文件上传处理:别再让用户等待了!

前端文件上传处理:别再让用户等待了! 毒舌时刻 文件上传?听起来就像是前端工程师为了显得自己很专业而特意搞的一套复杂流程。你以为随便加个input[type=file]就能实现文件上传?别做梦了!到时候你会发现,大文件上传会导致页面崩溃,用户体验极差。 你以为FormData就能解决所有问题?别天真了!FormData在处理大文件时会导致内存溢出,而且无法显示上传进度。还有那些所谓的文件上传库,看起来高大上,用起来却各种问题。 为什么你需要这个 1. 用户体验:良好的文件上传处理可以提高用户体验,减少用户等待时间。 2. 性能优化:合理的文件上传策略可以减少服务器负担,提高上传速度。 3. 错误处理:完善的错误处理可以避免上传失败时的用户困惑。 4. 安全保障:安全的文件上传处理可以防止恶意文件上传,保障系统安全。 5. 功能丰富:支持多文件上传、拖拽上传、进度显示等功能,满足不同场景的需求。 反面教材 // 1. 简单文件上传 <input type="file&

前端+AI:大厂前端岗位JD——总结前端AI学习路线

前端+AI:大厂前端岗位JD——总结前端AI学习路线

背景 现在AI技能是求职的默认必备技能,不管是传统的前后端项目还是现在AI潮流新涌出的AI应用开发工程师、AI Agent工程师以及最顶的AI 算法工程师,笔者为前端岗位,秋招投递了15+互联网大厂,收获3家大厂Offer(快手、京东、拼多多),下面聊聊个人对面试中的AI的一些idea: 1.大厂AI布局 2.透过JD看AI 总结(通过JD总结要学什么) 前端基础JS&算法、React&Vue框架、Vite、Monorepo、Pnpm工程化、性能优化、主流(微前端、SSR、大前端)仍为基础,全栈+AI是亮点,前端&AI学习: 一、前端AI体系架构总览 从这些JD可以看出,前端AI已经从“用AI做页面”升级为AI Native 前端,核心是: 前端开发者不再只是UI渲染层,

前端国际化:让你的网站走向世界

前端国际化:让你的网站走向世界 毒舌时刻 前端国际化?这不是大公司才需要的吗? "我的网站只面向国内用户,要什么国际化?"——结果业务拓展到海外,临时抱佛脚, "我直接用中文写死,多简单!"——结果需要支持英文时,满世界找字符串, "我用Google翻译,多快!"——结果翻译质量差,用户体验差。 醒醒吧,国际化不是可选的,而是现代前端开发的标配! 为什么你需要这个? * 全球用户覆盖:吸引来自不同国家和地区的用户 * 业务拓展:为未来的海外业务做准备 * 用户体验:让用户使用自己熟悉的语言 * 品牌形象:展现专业、全球化的品牌形象 反面教材 // 反面教材:硬编码字符串 function Header() { return ( <div className="header"> <