JS百题斩~秒懂数据的作用域(超详细)

2024-06-04 20:04

本文主要是介绍JS百题斩~秒懂数据的作用域(超详细),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

数据的作用域

定义:作用域是运行时代码中的变量,函数和对象的可访问性。通俗的意思就是数据在哪个范围是有效可用的,出了这个范围就不能用了。
作用域在哪,关键看在哪里定义的。
ES6之前没有块级作用域。

1.JS有两种作用域:全局作用域和函数作用域

全局作用域:在代码中任何地方都能访问到的对象拥有全局作用域。(所有未定义直接赋值的变量,自动声明为全局作用域)
函数作用域:函数作用域也叫局部作用域,指声明在函数内部的变量,函数的作用域一般只在固定代码片段中可访问到,例如函数内部。

全局作用域有弊端:如果我们写了很多行JS代码,变量定义都没有用函数包裹,那么他们就全部都在全局作用域中,这样就会污染全局命名空间,容易引起命名冲突。所以,很多第三方库的源码,所有的代码都在(function (){…})()立即执行函数中,这样保证里面的变量不会暴漏和外泄,污染全局变量,这就是函数作用域的作用。

  • 作用域是分层,里面的作用域能访问外面的,反之不行,访问的时候从内向外依次查找。
  • 如果在内部的作用域中访问了外部,则会形成闭包。
  • 内部作用域能访问的外部,取决于函数定义的位置,和调用无关,这就是所谓的静态作用域(作用域不等于执行上下文)。

2.变量提升

作用域里面定义的变量、函数声明会提升到作用域顶部。

3.块级作用域

ES5只有全局作用域和函数作用域,没有块级作用域,会带来以下问题:
1.变量提升导致内层变量可能会覆盖外层变量;

var  a = 5;
function f(){console.log(a);if(true){var a = 6;       }
}
f();
// undefined

2.for循环里面用来计数的变量泄露为全局变量;

// 需求:五个按钮,点击某个按钮, 提示'第 n 个'
for (let i = 0; i < btns.length; i++) {btns[i].onclick = function () {console.log('第' + (i + 1) + '个')}
}
// 结果:点击任何一个按钮,都是弹出'第 6 个'
// 这是因为 i 泄露为全局变量,执行到点击事件时,此时 i 的值为 5

所以ES6新增了块级作用域。
块级作用域可通过 let 和 const 声明,声明的变量在指定代码块的作用域外无法被访问。块级作用域在下面的情况下被创建:
1.函数内部;
2.代码块内部,用一对花括号包裹;

4.作用域链

当js中使用一个变量的时候,首先会尝试在当前作用域下去寻找该变量,如果没找到,再到它的上一层作用域找,一直找到该变量或者已经到了全局作用域,如果在全局作用域仍然找不到该变量,严格模式下会直接报错。
内部函数访问外部函数的变量,采取的是链式查找的方法来决定取哪个值,这种就称为作用域链。函数内部是根据就近原则来访问变量。

注意,变量的作用域,在创建时就已经确定好了,而非调用阶段确定的,所以某个作用于的上层作用域指的是包裹它的作用域,而非调用。

两个小练习:

// 下面的代码输出什么
var a = 1;
function f1() {a++;
}
function f2() {var a = 3;f1();console.log(a);
}
f2();
console.log(a);
// 3 2
// 内部作用域能访问的外部,取决于函数定义的位置,和调用无关。所以a++后打印的是2
// 下面的代码输出什么
var a = 1,b = 2;
function f() {console.log(a);var a = 2;function f1() {console.log(a, b);}f1();
}
f();
// undefined 3 2
// 作用域里面定义的变量、函数声明会提升到作用域顶部。所以打印第一个a时,变量a被提升到了f函数顶部,输出undefined

下一篇: JS百题斩~ typeof 与 instanceof 区别

这篇关于JS百题斩~秒懂数据的作用域(超详细)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1030980

相关文章

CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比

《CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比》CSS中的position属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布... css 中的 position 属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布局和层叠关

在Windows上使用qemu安装ubuntu24.04服务器的详细指南

《在Windows上使用qemu安装ubuntu24.04服务器的详细指南》本文介绍了在Windows上使用QEMU安装Ubuntu24.04的全流程:安装QEMU、准备ISO镜像、创建虚拟磁盘、配置... 目录1. 安装QEMU环境2. 准备Ubuntu 24.04镜像3. 启动QEMU安装Ubuntu4

SpringBoot整合Flowable实现工作流的详细流程

《SpringBoot整合Flowable实现工作流的详细流程》Flowable是一个使用Java编写的轻量级业务流程引擎,Flowable流程引擎可用于部署BPMN2.0流程定义,创建这些流程定义的... 目录1、流程引擎介绍2、创建项目3、画流程图4、开发接口4.1 Java 类梳理4.2 查看流程图4

SQL Server修改数据库名及物理数据文件名操作步骤

《SQLServer修改数据库名及物理数据文件名操作步骤》在SQLServer中重命名数据库是一个常见的操作,但需要确保用户具有足够的权限来执行此操作,:本文主要介绍SQLServer修改数据... 目录一、背景介绍二、操作步骤2.1 设置为单用户模式(断开连接)2.2 修改数据库名称2.3 查找逻辑文件名

SQL Server数据库死锁处理超详细攻略

《SQLServer数据库死锁处理超详细攻略》SQLServer作为主流数据库管理系统,在高并发场景下可能面临死锁问题,影响系统性能和稳定性,这篇文章主要给大家介绍了关于SQLServer数据库死... 目录一、引言二、查询 Sqlserver 中造成死锁的 SPID三、用内置函数查询执行信息1. sp_w

Python UV安装、升级、卸载详细步骤记录

《PythonUV安装、升级、卸载详细步骤记录》:本文主要介绍PythonUV安装、升级、卸载的详细步骤,uv是Astral推出的下一代Python包与项目管理器,主打单一可执行文件、极致性能... 目录安装检查升级设置自动补全卸载UV 命令总结 官方文档详见:https://docs.astral.sh/

Python包管理工具核心指令uvx举例详细解析

《Python包管理工具核心指令uvx举例详细解析》:本文主要介绍Python包管理工具核心指令uvx的相关资料,uvx是uv工具链中用于临时运行Python命令行工具的高效执行器,依托Rust实... 目录一、uvx 的定位与核心功能二、uvx 的典型应用场景三、uvx 与传统工具对比四、uvx 的技术实

canal实现mysql数据同步的详细过程

《canal实现mysql数据同步的详细过程》:本文主要介绍canal实现mysql数据同步的详细过程,本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的... 目录1、canal下载2、mysql同步用户创建和授权3、canal admin安装和启动4、canal

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte

Springboot3+将ID转为JSON字符串的详细配置方案

《Springboot3+将ID转为JSON字符串的详细配置方案》:本文主要介绍纯后端实现Long/BigIntegerID转为JSON字符串的详细配置方案,s基于SpringBoot3+和Spr... 目录1. 添加依赖2. 全局 Jackson 配置3. 精准控制(可选)4. OpenAPI (Spri