JS变量和函数提升

发布时间:2023年12月27日

JS变量提升

sayHi()

console.log(myname)

var myname = 'yy'

function sayHi() {
  console.log('Hi')
}

// 执行结果:
// Hi
// undefined

相信学过 JavaScript 的都知道这个执行结果的原理:JS的变量提升特性

注意:这里必须使用 var 来定义变量,如果使用 let 或者 const 来定义的变量不会有变量提升,会报错未定义。函数声明方式也具有变量提升特性。

所谓的变量提升,是指在 JavaScript 代码执行过程中,JavaScript 引擎把变量的声明部分和函数的声明部分提升到代码开头的“行为”。变量被提升后,会给变量设置默认值,这个默认值就是我们熟悉的 undefined

在这里插入图片描述

上图是我们模拟的变量提升的效果,从概念的字面意义上看变量和函数声明都会被移动到代码的最前面,但,这并不准确。实际上变量和函数声明在代码里的位置是不会改变的,一段 JavaScript 代码在执行之前需要被 JavaScript 引擎编译,编译完成后才会进入执行阶段。执行流程细化图如下

在这里插入图片描述

编译阶段

一段 JavaScript 代码,经过编译后,会生成两部分:执行上下文Execution context)和可执行代码

执行上下文是 JavaScript 执行一段代码时的运行环境,比如调用一个函数,就会进入这个函数的执行上下文,确定该函数在执行期间用到的诸如 this、变量、对象以及函数等。

在执行上下文中存在一个变量环境的对象Variable Environment),该对象中保存了变量提升的内容。下面来分析这个变量环境对象是如何生成的:

  1. 第1行和第2行,由于这两行都不是声明操作,所以 JavaScript 引擎不做任何处理
  2. 第3行,由于是经过 var 声明的,因此 JavaScript 引擎将在环境对象中创建一个名为 myname 的属性,并使用 undefined 对其初始化
  3. 第4行,JavaScript 引擎发现了一个通过 function 定义的函数,所以它将函数定义存储到堆(HEAP)中,并在环境对象中创建一个名为 sayHi 的属性,然后将该属性值指向堆中函数的位置。

就这样生成了变量环境对象。接下来 JavaScript 引擎会把声明以外的代码编译为字节码(可以类比可执行代码内容)。现在有了执行上下文和可执行代码,接下来就到了执行阶段了。

执行阶段

JavaScript 引擎会按照顺序一行一行地执行“可执行代码”,下面是整个执行过程:

  1. 当执行到 sayHi 函数时,JavaScript 引擎便开始在变量环境对象中查找该函数,由于变量环境对象中存在该函数的引用,所以JavaScript 引擎便开始执行该函数,打印 Hi 结果
  2. 接下来打印 myname 信息,JavaScript 引擎查找到该变量值在变量环境对象中的值为 undefined,打印 undefined 结果
  3. 接下来执行赋值操作,将 yy 赋值给 myname 变量,赋值后变量环境对象中的 myname 属性值变为 yy

相同变量或函数

那么如果代码中出现了重名的函数或者变量,JavaScript 引擎会如何处理呢?思考一下,如下代码输出的内容会是什么?

function sayHi() {
  console.log('Hi')
}

sayHi()

function sayHi() {
  console.log('Hello')
}

sayHi()

如果你理解了上面所说的 JavaScript 的执行机制:先编译,再执行。那么你一定能理解这段代码的输出结果

// Hello
// Hello
  1. 遇到第一个 sayHi 函数,会将函数定义存储到堆(HEAP)中,并在环境对象中创建一个名为 sayHi 的属性,然后将该属性值指向堆中函数的位置
  2. 遇到第二个 sayHi 函数,此时变量环境中已存在 sayHi 属性,此时将 sayHi 属性重新赋值指向堆中第二个函数的位置,这样变量环境中就只存在第二个函数了
  3. 执行第一次和执行第二次其实都是在执行最后的那个函数

再来看一个实例

sayHi() // Hello

var sayHi = function() {
  console.log('Hi')
}

// sayHi() // Hi

function sayHi() {
  console.log('Hello')
}
  1. 变量提升 sayHi 属性并初始化值为 undefined
  2. 函数提升 sayHi 属性指向 “Hello函数”
  3. 执行 sayHi() 函数等于执行 “Hello函数”,输出 Hello
  4. 变量 sayHi 赋值为 “Hi函数”

如果在 sayHi 函数声明前执行 sayHi() 函数,就会打印出 Hi,因为变量和函数提升后,就进行就是执行阶段,代码会一行一行往下执行,到这里时,sayHi 属性已经是 “Hi函数”了

模拟变量提升过程如下:

// 变量提升部分
var sayHi = undefined
function sayHi() {
  console.log('Hello')
}
// 可执行代码部分
sayHi() // Hello
sayHi = function() {
  console.log('Hi')
}
//sayHi() // Hi
文章来源:https://blog.csdn.net/weixin_43443341/article/details/135227768
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。