博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
详解js运行机制(1)-- 预解析
阅读量:6194 次
发布时间:2019-06-21

本文共 1434 字,大约阅读时间需要 4 分钟。

  一、问题的提出

  我们都知道,js是一个解释型的语言,js代码在运行时,是按照js在文档中出现的先后次序,依次逐条语句执行的。那么问题来了。我们看下面这个小例子

  这个程序能正确执行吗?

  如果按上面的理论,这个f1函数的调用出现在了声明的前面,显然当执行调用的语句时,js还没看到f1声明的部分,应该报错才对。但事实恰恰相反,这个程序是能正常执行的。这就牵涉到js程序执行的一个机制,叫做预解析。

  二、预解析定义

  其实js代码在执行过程分为两步,第一步叫预解析,第二步才是真正执行。

  所谓预解析,就是:在当前作用域中,JavaScript代码执行之前,浏览器首先会默认的把所有带var和function声明的变量进行提前的声明或者定义。

  对这句话的解析:

  1.何谓当前作用域,就是你声明的变量和函数其所在的作用域,在上例中,f1函数是声明在全局作用域下的,所以声明f1的当前作用域就是全局作用域。

  2.预解析不仅提升函数声明

  3.预解析还会提升变量声明,也谓之变量提升

  三、函数提升

  js执行环境会先扫描当前作用域中所有var声明的变量和function声明的函数,然后把他们提升到当前作用域的顶端。

  上例在运行时,f1的声明虽然是在调用语句之后,但其实js执行环境会对原始的代码做个预解析,解析过后的结果就变成下面这个样子

  差别很明显,一目了然,不解释。这样的代码当然能正常执行。

  同理,形如

  这里声明的2个函数f1和f2都会被提升,提升过后的结果是这样的:

  四、变量提升

  先看这个例子

  这个代码执行时会报错:,因为num没有声明。

  如果改成下面这个样子:

  结果是输出:undefined。以外吗?

  原因是js执行环境对var num = 10;这条声明变量的语句做了提升,它等价于  

  这就不难理解为啥输出的是undefined了。

  在看一个经典的面试题:

  这个例子输入的不是10,也不是20,正确答案是:undefined。

  原因:此题代码等价与

  在全局作用域下声明了一个变量num,在函数f1的作用域内声明了局部变量num,2个变量同名,那么在函数内部,起作用的是局部变量num,此时,后写的声明变量的语句var num = 20;被提升,当然它不会被提升到全局作用域,它只会被提升到声明这个局部变量的当前作用域的顶端,也就变成了等价代码的样子,所以输出就是undefined。

  结论:

  1.函数提升是整体提升

  2.变量提升是只提升声明,不提升赋值

  五、需要注意的地方

  (1)函数表达式不会被提升

  下面的例子执行时会报错

  错误如下:。错误提示是f1不是一个函数。

  为什么这里的f1没有被提升呢?因为f1是个函数表达式。

  如果深究,f1也被提升了,但是它没有被当成函数提升,而是当成变量了。因为这个时候f1是通过var声明的变量,只不过它指向了一个函数对象。所以它提升过后的结果等价于如下带代码

  这里f1被当成变量提升的,而f1又没有赋初值,所以f1在调用的时候还是undefined,那当然会报错了。

   (2)预解析不会跨越<script>代码块。函数和变量的提升,只在自己所在的<script>块之内提升。

转载于:https://www.cnblogs.com/ldq678/p/9758757.html

你可能感兴趣的文章
数据库设计原则
查看>>
es6中新增的常用数值扩展
查看>>
linux 不同网段pptp服务器的搭建
查看>>
45、【华为HCIE-Storage】--InfoTier
查看>>
打开Windows2008 R2 Server Manager或执行Windows Update时出现0x800F0818错误
查看>>
oracle set和col命令的整理
查看>>
perl类似于shell_date的功能
查看>>
centos6.4x64最小化安装部署rsync
查看>>
php中Maximum execution time of 120 seconds exceeded时间超时错误解决方案
查看>>
oracle 不可见索引
查看>>
查看Firefox 浏览器保存站点密码的方法
查看>>
8008.企业培训师课件--企业培训师总复习
查看>>
文件服务器综合测试(5-6题)
查看>>
BAP研究之bap_block_s
查看>>
文件查找和压缩
查看>>
raid+lvm+quota
查看>>
设置时间
查看>>
我的友情链接
查看>>
远程管理ServerCore服务器
查看>>
对自己档位定位
查看>>