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

FPGA逻辑设计仿真调试手把手教程

FPGA逻辑设计仿真调试实战全解析:从代码到波形的完整闭环 你有没有过这样的经历?写完一段Verilog代码,综合实现顺利通过,结果烧录进FPGA后功能完全不对。示波器一接,信号乱飞——可仿真时明明一切正常。这时候你会不会想:要是能直接“看”到芯片内部信号该多好? 别急,这正是我们今天要解决的问题。 在现代FPGA开发中, 仅靠写代码和烧板子已经远远不够了 。面对越来越复杂的数字系统,我们必须建立起一套科学、系统的验证方法论。本篇教程将带你走完从RTL编码到行为仿真、再到上板调试的全过程,手把手教你如何用Vivado构建一个真正可靠的FPGA开发流程。 从零开始:一个计数器背后的工程思维 我们先来看一个看似简单的例子: module counter_4bit ( input clk, input rst_n, output reg [3:0] count ); always @(posedge clk or negedge rst_n) begin if (!rst_n)

AI绘画模型格式转换完全指南:从问题诊断到场景化解决方案

AI绘画模型格式转换完全指南:从问题诊断到场景化解决方案 【免费下载链接】awesome-ai-paintingAI绘画资料合集(包含国内外可使用平台、使用教程、参数教程、部署教程、业界新闻等等) stable diffusion tutorial、disco diffusion tutorial、 AI Platform 项目地址: https://gitcode.com/GitHub_Trending/aw/awesome-ai-painting 在AI绘画工作流中,模型格式转换是连接不同工具链的关键环节。当你遇到"无法加载模型文件"的错误提示,或是发现存储空间被低效格式占用时,掌握模型格式转换技术就成为解决问题的核心能力。本文将通过诊断指南的形式,帮助你系统理解模型格式的选择策略、实施转换流程、验证转换效果,并探索在不同场景下的应用方案,让你的AI绘画工作流更加高效与稳定。 问题诊断:你的模型格式是否需要优化? 格式兼容性故障排查 当你的AI绘画工具弹出"无法加载CKPT文件"的错误时,首先需要判断这是否是格式兼容性问题。常见的症状包括:

一文读懂“JESD204B”之链路建立与xilinx IP仿真

一、初识 JESD204B 是由JEDEC(电子器件工程联合会)制定的高速串行接口标准,主要用于数据转换器(ADC/DAC)与数字处理器(如FPGA、ASIC)之间的数据传输。在JESD标准出来前,常用的是传统的LVDS接口:LVDS(Low-Voltage Differential Signaling,低压差分信号)是一种广泛应用的物理层电气标准,用于高速、低功耗的差分信号传输,但是在使用LVDS接口时,对阻抗和多通道时延要求比较严格,因为LVDS使用的是源同步接口,允许时钟和多个数据通道同时传输,时钟信号和数据保持确定的相位关系,同时由发送端(图中的外部器件)传输至接收端(比如FPGA)。接收端利用对端传送来的时钟信号作为采样时钟,对数据位进行采样。在采样过程中,只要保证接收端时钟信号与接收数据满足一定的建立/保持时间,数据即可被正确接收。 图 1 源同步LVDS接口 因此我们可以知道,LVDS对各通道的时延要求是比较高的,因此PCB布线要求也比较严格,差分对需阻抗匹配(100Ω±10%)和等长控制(长度差<

【PZ-VU9P & PZ-VU13P】璞致FPGA开发板:Xilinx Virtex UltraScale Plus核心板与开发板深度解析

1. 璞致FPGA开发板与Xilinx Virtex UltraScale Plus架构解析 第一次拿到璞致PZ-VU9P开发板时,就被它沉甸甸的金属散热片震撼到了。这可不是普通的FPGA开发板,而是搭载Xilinx旗舰级Virtex UltraScale Plus芯片的"性能怪兽"。先说说这个16nm工艺的Virtex UltraScale Plus架构,它就像是FPGA界的"超级跑车引擎"——在计算密集型应用中,既能飙出26Gbps的GTY收发器速度,又能通过3D-on-3D芯片堆叠技术实现惊人的能效比。 实测在图像处理项目中,VU13P的378万个逻辑单元可以同时处理4路8K视频流,而功耗仅为上代产品的70%。这种性能突破主要来自三大黑科技: * UltraRAM:片上集成432Mb超大容量存储,相当于给数据修了条"高速公路匝道",避免频繁访问外部DDR造成的拥堵 * CLB架构升级:每个可配置逻辑块(CLB)包含8个查找表+16个触发器,布线资源增加40%,我在做波束成形算法时实测布线成功率提升明显 * DSP48E2切片:支持27x18乘法运算和48位累加,做矩阵