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

相关文章

MySQL 删除数据详解(最新整理)

《MySQL删除数据详解(最新整理)》:本文主要介绍MySQL删除数据的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、前言二、mysql 中的三种删除方式1.DELETE语句✅ 基本语法: 示例:2.TRUNCATE语句✅ 基本语

Python设置Cookie永不超时的详细指南

《Python设置Cookie永不超时的详细指南》Cookie是一种存储在用户浏览器中的小型数据片段,用于记录用户的登录状态、偏好设置等信息,下面小编就来和大家详细讲讲Python如何设置Cookie... 目录一、Cookie的作用与重要性二、Cookie过期的原因三、实现Cookie永不超时的方法(一)

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

SpringBoot整合liteflow的详细过程

《SpringBoot整合liteflow的详细过程》:本文主要介绍SpringBoot整合liteflow的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋...  liteflow 是什么? 能做什么?总之一句话:能帮你规范写代码逻辑 ,编排并解耦业务逻辑,代码

MyBatisPlus如何优化千万级数据的CRUD

《MyBatisPlus如何优化千万级数据的CRUD》最近负责的一个项目,数据库表量级破千万,每次执行CRUD都像走钢丝,稍有不慎就引起数据库报警,本文就结合这个项目的实战经验,聊聊MyBatisPl... 目录背景一、MyBATis Plus 简介二、千万级数据的挑战三、优化 CRUD 的关键策略1. 查

python实现对数据公钥加密与私钥解密

《python实现对数据公钥加密与私钥解密》这篇文章主要为大家详细介绍了如何使用python实现对数据公钥加密与私钥解密,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录公钥私钥的生成使用公钥加密使用私钥解密公钥私钥的生成这一部分,使用python生成公钥与私钥,然后保存在两个文

mysql中的数据目录用法及说明

《mysql中的数据目录用法及说明》:本文主要介绍mysql中的数据目录用法及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、版本3、数据目录4、总结1、背景安装mysql之后,在安装目录下会有一个data目录,我们创建的数据库、创建的表、插入的

浏览器插件cursor实现自动注册、续杯的详细过程

《浏览器插件cursor实现自动注册、续杯的详细过程》Cursor简易注册助手脚本通过自动化邮箱填写和验证码获取流程,大大简化了Cursor的注册过程,它不仅提高了注册效率,还通过友好的用户界面和详细... 目录前言功能概述使用方法安装脚本使用流程邮箱输入页面验证码页面实战演示技术实现核心功能实现1. 随机

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C

Navicat数据表的数据添加,删除及使用sql完成数据的添加过程

《Navicat数据表的数据添加,删除及使用sql完成数据的添加过程》:本文主要介绍Navicat数据表的数据添加,删除及使用sql完成数据的添加过程,具有很好的参考价值,希望对大家有所帮助,如有... 目录Navicat数据表数据添加,删除及使用sql完成数据添加选中操作的表则出现如下界面,查看左下角从左