JS的执行上下文,变量声明提升,函数声明提升

2023-10-27

一、什么是执行上下文(execution content)

一句话:执行上下文就是当前JS代码被解析和执行时存在的环境。(ECMAScript中定义的抽象概念)

二、执行上下文的类型

  1. 全局执行上下文:这是默认的,最基础的执行上下文,只有一个
  • 不在函数内部的代码都位于全局执行上下文中

  • 创建一个全局对象,其实就是我们的window对象

  • 将this指向这个全局对象

  1. 函数执行上下文:每次函数被调用的时候,就会创建一个新的执行上下文
  • 每个函数都会有自己的执行上下文
  • 一个程序中可以存在任意数量的函数执行上下文
  • 每一个函数执行上下文被创建的时候,它都会按照特定的顺序执行一系列的步骤。
  1. eval函数执行上下文:运行在eval函数中的代码会获得自己的执行上下文。(很少用到,不做讨论)

三、执行上下文的生命周期(以函数执行上下文为例)

创建阶段 => 执行阶段 => 回收阶段

  1. 创建阶段

当函数被调用,但是未执行内部的任何代码之前。

这个阶段会做出以下几件事:

  • 创建变量对象:首先会初始化函数的参数arguments,提升函数声明和变量声明。

  • 创建作用域链:在执行上下文创建阶段,作用域链是在变量对象之后创建的,作用域链本身包含变量对象,作用域链用来解析变量。

  • 确定this指向

    • function函数是在被调用时确定this指向
    • 箭头函数在声明时就已经确定了this指向,箭头函数的this由其所处的上下文决定
  1. 执行阶段
  • 变量赋值,执行代码
  1. 回收阶段
  • 执行上下文出栈,JS自动执行垃圾回收机制。

四、变量声明提升

我们看下边这段代码:

在变量还没有声明的时候进行打印,会打印出什么呢?

<script>
       console.log(a);
       var a = 10;
</script>

结果:
在这里插入图片描述
我们看到不但没有报错 Uncaught ReferenceError: Cannot access ‘a’ before initialization(变量a未定义),还打印出了undefined。

事实上,在执行代码之前,我们的浏览器会先解析一遍我们的脚本,完成一个初始化的步骤,它遇到var变量时就会先初始化变量为undefined。

这就是变量提升(hoisting),它是指,浏览器在遇到JS执行环境的初始化时,引起的变量提前定义。

如何避免变量提升呢?

使用let,const 关键字声明变量,尽量使用从上图,避免使用var

<script>
       console.log(a);//报错:Uncaught ReferenceError: Cannot access 'a' before initialization
       let  a = 10;

       /*--------------------分割线---------------*/

       console.log(b);//报错:Uncaught ReferenceError: Cannot access 'a' before initialization
       const b = 20;
    </script>

五、函数声明提升

  • 声明函数的方法(箭头函数属于第二种声明方式)

    • 函数声明方式:function foo(){}
    • 函数表达式方式:var foo = function(){}或var foo = ()=>{}

我们一起看一段代码:

 <script>
    console.log(f1); //ƒ f1(){}
    function f1(){}
    /*--------------分界线-------------*/
    console.log(f2);//undefined
    var f2 = function(){}
 </script>

同样我们看到,在声明函数之前打印该函数,打印出了内容,这是因为函数提升,就是初始化时,引擎把函数声明整个提升到了当前作用域的顶部。在实际打印之前,函数已经被声明。

至于第一个与第二个函数打印结果不同,我们结合第四点,变量提升,第二个函数使用函数表达式的方式声明,将函数赋值给了一个变量,这个变量在提升时被初始化为undefined。

而第一个函数使用函数声明方式声明,打印的是该函数本身。

注意:

  • 当函数和变量同名时,函数声明的优先级高于变量声明的优先级,因此变量声明会被函数声明给覆盖掉,但是可以重新赋值。
<script>
	alert(a)//弹出function a(){alert('我是函数');}
	var a = "我是变量"
	function a(){alert('我是函数');}
</script>
  • 如果在同一个作用域中存在多个同名函数声明,后面出现的将会覆盖前面的函数声明。
function hoistFunction() {
    function foo() {
        console.log(1);
    }

    foo(); // 2

    function foo() {
        console.log(2);
    }
}

hoistFunction();
  • 函数声明比函数表达式声明优先级高。
function hoistFunction() {
            foo() // 2
            
            var foo = function () {
                console.log(1);
            };

            foo() // 1

            function foo() {
                console.log(2);
            }

            foo(); // 1
        }

        hoistFunction();
//解释JS中,函数是一等公民,函数声明的优先级最高,会被提升到当前作用域的最顶端,所以第一次调用时实际执行了下面的函数声明,第二次调用时,由于前边的函数表达式与之前的函数声明同名,故将其覆盖,以后的调用也将会打印出相同的结果。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

JS的执行上下文,变量声明提升,函数声明提升 的相关文章

随机推荐

  • 网络基础:路由器工作原理

    目录 一 理论 1 路由器工作原理 2 路由表获取方式 3 路由器和交换机的区别 二 实验 1 华为系统配置2台路由器 2 华为系统配置3台路由器 3 华为系统配置浮动路由 一 理论 1 路由器工作原理 路由器工作在OSI七层协议中的第三层
  • AD画PCB时如何修改编辑区(黑色部分)

    今天同事提出这个问题 心中暗喜 又到我发光的时候了 结果 WHAT 没研究明白 小姐姐明明之前会N种方法的呀 赶紧问度娘 问题解决 应该是ad不同版本指令不太一样了 方法一 直接在KEEPOUT层画出想要的板框大小 方向 选中状态 执行DE
  • linux C错误汇集

    问题一 22 c In function main 22 c 8 9 error empty character constant 解决方法 少了空格 问题2 没有连接到math函数库 gcc 12 c o 12 lm 问题3 segmen
  • 如何在Cocos2d-x中集成安卓版微信SDK

    本文转载于 http www cocoachina com bbs read php tid 224616 1 概述 微信的火热程度已不必说了 如果能在Cocos2d x中集成微信SDK 让游戏能分享给好友或者朋友圈 那对游戏推广将会是非常
  • QT引入及基本介绍

    文章目录 嵌入式项目的图形解决方案 QT 跨平台的实现 qmake vs cmake base class QWidget QMainWindow QDialog kits Hello World 创建项目 hello pro项目文件 ui
  • img标签及属性

    一 img标签 1 img标签的作用 用来告诉浏览器我们需要显示一张图片 2 img标签格式 img src img标签中的src是英文source的缩写 src 是用来告诉img标签 需要显示的图片名称 3 注意点 和H系列标签 P标签
  • C语言每日一练——第133天:打鱼还是晒网

    前言 Wassup guys 我是Edison 今天是C语言每日一练 第133天 Let s get it 文章目录 1 问题描述 2 题目分析 3 算法设计 4 流程框架 求出指定日期距离 5 代码实现 1 问题描述 中国有句俗语叫 三天
  • java中统计一个字符串中出现最多的字符和次数

    用Map来存储被拆分成数组的字符串 key为字符 value为出现次数 package net hncu other import java io UnsupportedEncodingException import java util
  • Unity快速适配IOS/安卓刘海屏(又简单又快 适配了O版本和P版本)

    刘海屏适配 其实就是知道刘海高度 横屏游戏 来对ui进行偏移 所以刘海屏适配的关键是获取刘海高度 获取刘海高度有三种方案 1 大数据 收集各种型号对应的刘海数据 听说腾讯有些项目这么搞 2 代码获取 热门机型获取刘海数据 小众机型不是and
  • XGBoost详解

    文章目录 背景 目标函数 最优切分点算法 Shrinkage 收缩过程 缺失值处理 优缺点 总结 背景 在看Xgboost之前 先看看笔者写的AdaBoost 和GBDT AdaBoost 关注的是哪些错误分类的样本 每次加大误分类样本的权
  • 越丰满的稻穗,头垂得越低

    author skate time 2010 06 18 职场中有三种人 主动 空杯 者 被动 空杯 者和拒绝 空杯 者 毫无疑问 拒绝 空杯 者 要么停滞不前 要么倒退 要么成为他人的绊脚石 一般情况下 人们会把自己没有获得职业升迁的责任
  • java自动化测试语言基础之Stream、File和IO

    java自动化测试语言基础之Stream File和IO 文章目录 java自动化测试语言基础之Stream File和IO Java 流 Stream 文件 File 和IO Java 流 Stream 文件 File 和IO Java
  • Flutter FutureBuilder 返回空 Flutter_BUG_A build function returned null

    一定是没有返回widget 检查下return了没有
  • vs2019+QT新建UI窗口

    在vs2019中新建一个QT的widget窗口 第一步 第二步 其中可以选择MainWindow或者Widget窗口 填写好ui的名字 第三步 建立好对应ui名字的 h 和 cpp文件即可 双击新建的ui文件 如果打不开的话 可以重新配置一
  • 解决nes_py在pip安装报错的问题

    目录 项目场景 问题描述 原因分析 解决方案 解决结果 项目场景 想跟随油管某视频复现强化学习方法玩超级马里奥的过程 结果在在Anaconda3虚拟环境中用pip安装nes py时一直报错 报错信息如下 Building wheel for
  • 浏览器与Node的事件循环(Event Loop)有何区别?

    前言 本文我们将会介绍 JS 实现异步的原理 并且了解了在浏览器和 Node 中 Event Loop 其实是不相同的 一 线程与进程 1 概念 我们经常说JS 是单线程执行的 指的是一个进程里只有一个主线程 那到底什么是线程 什么是进程
  • PAT乙级1043 输出PATest (20 分)

    1043 输出PATest 20 分 一 问题描述 给定一个长度不超过 10 4 的 仅由英文字母构成的字符串 请将字符重新调整顺序 按 PATestPATest 这样的顺序输出 并忽略其它字符 当然 六种字符的个数不一定是一样多的 若某种
  • 初识微服务技术栈

    目录 什么是微服务 注册中心 配置中心 服务网关 分布式缓存和数据库 分布式搜素 消息队列 分布式日志服务 系统的监控链路追踪 持续集成 1 认识微服务 1 0 学习目标 1 1 单体架构 1 2 分布式架构 1 3 微服务 1 4 微服务
  • 由于找不到packet.dll,无法继续执行代码的多种解决方法分享

    在计算机领域中 packet dll是一个重要的动态链接库文件 它被用来进行网络数据包的捕获和分析 然而 有时我们可能会遇到packet dll缺失的问题 这将导致我们无法正常执行代码 下面我们将为你详细介绍如何解决这个问题 以确保你的代码
  • JS的执行上下文,变量声明提升,函数声明提升

    目录 一 什么是执行上下文 execution content 二 执行上下文的类型 三 执行上下文的生命周期 以函数执行上下文为例 四 变量声明提升 五 函数声明提升 一 什么是执行上下文 execution content 一句话 执行