变量提升和函数提升的总结

  我们在学习JavaScript时,会遇到变量提升和函数提升的问题,为了理清这个问题,现做总结如下,希望对初学者能有所帮助

  我们都知道 var 声明的变量有变量提升,而 let 声明的变量和 const 声明的常量没有变量提升,但是为什么 let 声明的变量就不会变量提升呢?

首先我们得理解变量和函数的创建过程:

  变量声明 ==>变量初始化 ==>变量赋值

  函数声明 ==> 初始化 ==> 赋值

而变量提升和函数提升要完成的工作为:

  变量声明 ==> 变量初识化(undefined)

  函数声明 ==> 初始化 ==> 赋值(仅仅是赋值,并没有运行函数体)

暂时性死区的概念:

  ES6规定,如果代码区块中存在 let 和 const 命令声明的变量,这个区块对这些变量从一开始就形成了封闭作用域,直到声明语句完成,这些变量才能被访问(获取或设置),否则就会报错 ReferenceError。这在语法上称为“暂时性死区”,即代码块开始到变量声明语句完成之间的区域。通过 var 声明的变量拥有变量提升,没有暂时性死区。

  因为有暂时性死区的存在,所以 let 声明的变量和 const 声明的常量才 不能 变量提升

  

  从变量创建过程来理解:

  let 声明被提升了,但是并没有做初始化工作

  const 声明被提升,也是没有做初始化工作(const只有声明和初始化过程,没有赋值过程)

    let 和 const 都没有提升所要满足的工作,所以就无法提升

  var 的声明被提升,初始化被提升

  function 的声明、初始化、和赋值都被提升

var 声明的变量提升(声明、初始化都被提升,但是赋值没有提升)

  实例  

  console.log(a)  // undefined

  var a = 'hello world'

  console.log(a)  // 'hello world'

  实际执行过程如下

  var a

  console.log(a)  // undefined

  a = 'hello world'

  console.log(a)  // 'hello world'

function 声明的函数提升(声明、初始化、赋值都提升了)

  实例  

  fn()  // 'hello world'

  function fn() { console.log('hello world') }

  fn()  // 'hello world'

  实际执行过程如下

  function fn() { console.log('hello world') }  // 注意:此时函数体并没有运行

  fn()  // 'hello world'

  fn()  // 'hello world'

既有 var 声明的变量提升,又有 function 声明的函数提升

  实例

  变量名和函数名不同名  

  console.log(a)  // undefined

  fn()

  var a = 100

  console.log(a)  // 100

  function fn() {

    console.log(a)  // undefined

    var a = 10

    console.log(a)  // 10

   }

  var a = 200

  console.log(a)  // 200

  实际执行过程如下 

  function fn() {

    console.log(a)  

    var a = 10

    console.log(a)   

   }

  var a  (全局变量)

  var a  (局部变量)

  console.log(a)  // undefined

  fn()  // undefined  10

  a = 100

  console.log(a)  // 100

  a = 200

  console.log(a)  // 200

  变量名和函数名同名 

  情况一:变量由 函数类型 变为 基本数据类型(如 Number)

  var a = 100

  console.log(a)  // 100

  a()  // 报错(TypeError:a is not a function)

  function a() {

    console.log('hello world') 

   }

  实际执行过程如下

  function a() {

    console.log('hello world') 

   }

  var a  // 用 var 重新声明了变量 a,但此时变量 a 指向的还是函数体本身,值未改变

  a = 100  //  重新给 a 赋值为 100,a 的值发生了改变,由函数类型变为了number类型

  console.log(a)  // 100

  a()  // 报错:此时 a 已经不是函数 a 了

 

  情况二:变量重新赋值,但未立即改变变量的值

  console.log(a)  // a(){ console.log('hello world')}  即 a 函数本身

  a()  // 'hello world'

  var a = 100

  var a = 200

  var a = 300

  console.log(a)  // 300

  function a() {

    console.log('hello world') 

   }

  实际执行过程如下

  function a() {

    console.log('hello world')

   }

  var a  //  此时虽然重新声明了 a 变量,但是并没有改变 a 的值,此时的 a 还是个函数

  console.log(a)  // a(){ console.log('hello world')}  即 a 函数本身

  a = 100  // 重新给 a 变量赋值 100

  a = 200  // 重新给 a 变量赋值 200

  a = 300  // 重新给 a 变量赋值 300

  console.log(a)  // 100(此时变量 a 已经由 函数类型变为 基本数据类型number,且值为 100)

  a()  // 报错:此时 a 已经不是函数 a 了

用 let 定义的变量和用 const 定义的常量 没有提升

用 let 定义的变量

  在变量定义之前使用改变量 会报错

  console.log(a)  // 报错  Cannot access 'a' before initialization

  let a = 10

  

  在变量定义之后使用改变量 才能正常运行

  let a = 10  

  console.log(a)  // 10

总结:

  1. 用 var 定义的变量有变量提升,用 function 定义的函数有函数提升

  2. 函数提升优先级高于变量提升,即优先执行函数提升

  3. let 定义的变量和 const 定义的常量不会提升

  4. 重复用 var 定义的同一个变量是合法的,如果重复的声明有初始值,那就赋值;如果没有初始值,那么它不会对变量原来保存的值不会产生影响

  5. 不能用 let 重复定义同一个变量,会造成错误

js中变量提升和函数提升的更多相关文章

  1. JS中变量名和函数名重名

    今天骚凯问了一道变量名冲突的题目,感觉很有意思,顺便也复习一下预解析的一些知识,有不对的地方忘前辈大神指正,题目是这样的: var a=100; function a(){ console.log(a ...

  2. js中的变量提升和函数提升

    从上周开始,我所在的学习小组正式开始了angular的学习,angular是全面支持es6的,所以语法上和以前的angular有了很大的不同,比如变量声明时就抛弃了var,而选择了let和const: ...

  3. 关于JS中变量提升的规则和原理的一点理解

        关于变量提升,以前在一些教程和书籍上都听到过,平时开发中也知道有这个规律,但是今天突然在一个公开课中听到时,第一反应时一脸懵逼,然后一百度,瞬间觉得好熟悉啊,差点被这个概念给唬住了,不信我给你 ...

  4. JS 变量提升与函数提升

    JS 变量提升与函数提升 JS变量提升 变量提升是指:使用var声明变量时,JS会将变量提升到所处作用域的顶部.举个简单的例子: 示例1 console.log(foo); // undefined ...

  5. js变量提升与函数提升的详细过程

    大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...

  6. JS——变量提升和函数提升

    一.引入 在了解这个知识点之前,我们先来看看下面的代码,控制台都会输出什么 var foo = 1; function bar() { if (!foo) { var foo = 10; } aler ...

  7. 关于JS中变量提升的规则和原理的一点理解(二)

    上篇文章中讲到变量提升和函数提升的先后顺序时蒙了,后来去查了一下资料,特别整理一下. 在<你不知道的JavaScript(上卷)>一书的第40页中写到:函数会首先被提升,然后才是变量. 书 ...

  8. 谈谈javascript中的变量提升还有函数提升

    在很多面试题中,经常会看到关于变量提升,还有函数提升的题目,所以我就写一篇自己理解之后的随笔,方便之后的查阅和复习. 首先举个例子 foo();//undefined function foo(){ ...

  9. JS逻辑题 技术点: 1). 变量提升 2). 函数提升 3). 预处理 4). 调用顺序

    考查的技术点:  1). 变量提升 2). 函数提升  3). 预处理  4). 调用顺序 var c = 1; function c(c) { console.log(c); var c = 3; ...

随机推荐

  1. Air530Z GPS/北斗定位模块_设计指导手册_V1.2

    下载PDF版本: Air530Z_定位模块_设计指导手册_V1.2.pdf @ 目录 1. 模块整体说明 2. 资料下载 3. 模块性能 4.模块管脚图 5.参考设计电路 6.GPS天线 6.1 无源 ...

  2. NOIP模拟测试2「排列 (搜索)·APIO划艇」

    排序 内存限制:128 MiB 时间限制:1000 ms 标准输入输出     题目描述 输入格式 数据范围与提示 对于30%的数据,1<=N<=4: 对于全部的数据,1<=N< ...

  3. 我用段子讲.NET之依赖注入其一

    <我用段子讲.NET之依赖注入其一> 1) 西城的某个人工湖畔,湖水清澈见底,湖畔柳树成荫.人工湖往北,坐落着两幢写字楼,水晶大厦靠近地铁站,由于为了与湖面天际线保持一致,楼层只有26层高 ...

  4. R-常见错误

    错误一:选择了未定义的列(Undefined columns are selected) 改正方法:把目标列转换成因子类型(as.factor) 使用代码如下: 或者: 错误二:太多(36119)的重 ...

  5. PyCharm 2020.1 激活教程

    本文内容皆为作者原创,如需转载,请注明出处:https://www.cnblogs.com/xuexianqi/p/14967434.html 免责声明:本方法只做学习研究之用,不得用于商业用途 若经 ...

  6. hdu 6050 Funny Function 矩阵快速幂

    就算告诉我是矩阵快速幂我也推不出递推式呀!!! 官方题解: 对于任意i>=1,当j>=3时,有通过归纳法可以得到 进而推导出 后来自己重新推导了一遍 #include <iostre ...

  7. [Kick Start] 2021 Round B

    题目:Kick Start 2021 Round-B . Increasing Substring 输出字符串中每个字符的最长 Increasing Substring 的长度,非常简单的动态规划问题 ...

  8. CentOS-Docker安装RabbitMQ(单点)

    这里注意获取镜像的时候要获取management版本的,不要获取last版本的,management版本的才带有管理界面. 获取镜像 $ docker pull rabbitmq:management ...

  9. JVM,我就不信学不会你了

    JVM 对 Java 有多重要,对程序员面试有多重要,这些不用多说. 如果你还没意识到学 JVM 的必要性,或者不知道怎么学 JVM,那么看完这篇文章,你就能知道答案了. 曾经的我很不屑于学 JVM, ...

  10. 开始前端三大基础的js之途

                                                     初识   js                                             ...