JavaScript 中的词法作用域与 let 变量遮蔽机制详解

发布时间 - 2026-02-02 00:00:00    点击率:

本文解释为何在嵌套函数中重复声明 `let x` 不会报错——因为这是合法的变量遮蔽(shadowing),而非重复声明;`let` 的“不可重复声明”限制仅适用于同一作用域内。

在 JavaScript 中,let(以及 const)具有块级作用域(block scope),其核心规则之一是:同一作用域内不允许重复声明同名变量。但关键在于——“同一作用域”指的是完全相同的词法作用域层级,而非“名称相同”或“嵌套关系中看似冲突”。

回到你的示例代码:

function demoFunction() {
  let x = 10; // ← 外层函数作用域中的 x

  function demo() {
    let y = 20;
    let x = 5; // ← 内层函数作用域中的 x —— 全新变量,非重声明!
    console.log(x); // 输出 5
    console.log(y); // 输出 20
  }

  console.log(x); // 输出 10
  demo();
}

demoFunction(); // 输出:10 → 5 → 20

这里并未违反 let 规则,因为:

  • 外层 let x = 10 属于 demoFunction 函数作用域;
  • 内层 let x = 5 属于 demo 函数作用域;
  • 两个 x 位于不同、互不重叠的词法作用域中,因此它们是两个独立变量。

这种行为称为 变量遮蔽(variable shadowing):内层作用域中声明的同名变量会“遮蔽”外层同名变量,在该内层作用域内对 x 的所有引用均指向内层变量,而外层 x 依然存

在且不受影响。

立即学习“Java免费学习笔记(深入)”;

⚠️ 对比 var 的差异:
var 仅支持函数作用域,且存在变量提升(hoisting)和重复声明容忍(虽不推荐)。例如:

function example() {
  var x = 10;
  if (true) {
    var x = 20; // ✅ 合法(但会覆盖外层 x,无块级隔离)
  }
  console.log(x); // 输出 20
}

而 let 在块级(如 if、for、函数体)中严格隔离,且禁止同一块内重复声明:

function badExample() {
  let x = 10;
  let x = 20; // ❌ SyntaxError: Identifier 'x' has already been declared
}

✅ 正确理解作用域链:
当访问一个变量时,JavaScript 引擎按从内到外的嵌套顺序查找最近的声明(即作用域链):

let x = 'global';

function outer() {
  let x = 'outer';
  function inner() {
    let x = 'inner';
    console.log(x);        // 'inner' ← 最近声明
  }
  inner();
}
outer();

总结:

  • ✅ 允许在不同作用域中用 let 声明同名变量(即遮蔽);
  • ❌ 禁止在同一作用域(如同一函数体、同一 {} 块)中重复声明 let 变量;
  • ? 作用域由词法结构(函数、块)静态决定,与执行时机无关;
  • ? 遮蔽是显式设计特性,常用于避免意外修改外层变量,提升代码可维护性。

掌握这一机制,能帮你写出更安全、意图更清晰的现代 JavaScript 代码。


# javascript  # java  # win  # 作用域  # red  # if  # for  # const  # var  # 变量提升  # 而非  # 这是  # 这一  # 一是  # 帮你  # 适用于  # 不受  # 报错  # 指的是  # 虽不 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 电视网站制作tvbox接口,云海电视怎样自定义添加电视源?  Laravel怎么实现验证码功能_Laravel集成验证码库防止机器人注册  利用JavaScript实现拖拽改变元素大小  手机软键盘弹出时影响布局的解决方法  高防服务器租用指南:配置选择与快速部署攻略  laravel怎么配置和使用PHP-FPM来优化性能_laravel PHP-FPM配置与性能优化方法  Laravel中间件如何使用_Laravel自定义中间件实现权限控制  如何基于云服务器快速搭建网站及云盘系统?  Edge浏览器提示“由你的组织管理”怎么解决_去除浏览器托管提示【修复】  如何确保FTP站点访问权限与数据传输安全?  深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?  Win11搜索栏无法输入_解决Win11开始菜单搜索没反应问题【技巧】  laravel怎么通过契约(Contracts)编程_laravel契约(Contracts)编程方法  php json中文编码为null的解决办法  ,怎么在广州志愿者网站注册?  如何在宝塔面板中创建新站点?  Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)  公司网站制作需要多少钱,找人做公司网站需要多少钱?  Win11怎么开启自动HDR画质_Windows11显示设置HDR选项  如何快速搭建个人网站并优化SEO?  Android滚轮选择时间控件使用详解  广州网站制作公司哪家好一点,广州欧莱雅百库网络科技有限公司官网?  如何快速上传自定义模板至建站之星?  如何自定义建站之星网站的导航菜单样式?  如何确认建站备案号应放置的具体位置?  网站制作大概要多少钱一个,做一个平台网站大概多少钱?  Laravel中Service Container是做什么的_Laravel服务容器与依赖注入核心概念解析  如何快速配置高效服务器建站软件?  Laravel如何实现事件和监听器?(Event & Listener实战)  再谈Python中的字符串与字符编码(推荐)  Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  佐糖AI抠图怎样调整抠图精度_佐糖AI精度调整与放大细化操作【攻略】  个人网站制作流程图片大全,个人网站如何注销?  Laravel如何使用Guzzle调用外部接口_Laravel发起HTTP请求与JSON数据解析【详解】  1688铺货到淘宝怎么操作 1688一键铺货到自己店铺详细步骤  Windows10电脑怎么查看硬盘通电时间_Win10使用工具检测磁盘健康  Win11怎么关闭专注助手 Win11关闭免打扰模式设置【操作】  香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化  Win11摄像头无法使用怎么办_Win11相机隐私权限开启教程【详解】  如何在Windows服务器上快速搭建网站?  C++用Dijkstra(迪杰斯特拉)算法求最短路径  家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?  如何在IIS管理器中快速创建并配置网站?  软银砸40亿美元收购DigitalBridge 强化AI资料中心布局  网站制作企业,网站的banner和导航栏是指什么?  如何做网站制作流程,*游戏网站怎么搭建?  如何快速重置建站主机并恢复默认配置?  Mybatis 中的insertOrUpdate操作  Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】