1.1什么是函数提升和变量的提升?

JS引擎在运行整个JS代码的过程中,分为俩步。

第一步是读取和解析JS代码,第二部是执行。

在引擎解析JS代码的时候,当解析器遇见变量声明(var 变量名)和函数声明

(function 函数名)的时候,会将这些声明提到各自作用域的最前面。

1.2 作用域

在ES6之前,JS是没有块级作用域的。只有2种作用域:

  • 1.全局作用域
  • 2.函数作用域

注:在其他语言中,{...}被称为代码块,所形成的作用域被称作块级作用域,如:

  1. if(...){
  2. var a = 1;
  3. }
  4. for(var i = 0;i<5;i++){
  5. var b = 1;
  6. }
  7. console.log(a);
  8. console.log(b);
  9. console.log(i);

如果以上俩例子有块级作用域的话,a,b,i是不能被访问到的,但是

在JS中是可以的(ES6之前),所以没有块级作用域。

2.1 变量提升

2.1.1 声明变量

首先我们声明变量是通过var关键字,如果不用var直接赋值的话会被解析器

当作是全局变量。但是切记,变量的提升只限于用var声明的变量,没通过

var声明的变量将不会被提升。看下边的例子:

  1. var a = 1;
  2. console.log(a);// 1
  3. // 输出的结果为1,没问题,因为代码在执行的时候是按顺序执行的。
  4. // 解析的时候,经过变量提升后的结果如下:
  5. var a;
  6. a = 1;
  7. console.log(a);// 1
  1. console.log(a);// undefined
  2. var a = 1;
  3. // 这次的结果输出为undefined,我们都知道,当一个变量声明了却未赋值的时候就会出现undefined,
  4. // 但是一个变量没有声明的话,就会输出 a is not defined,先看本例经变量提升后的结果:
  5. var a ;
  6. console.log(a);// undefined
  7. a = 1;
  8. // 所以由此可见,变量的提升只会提升变量的声明,而不会提升变量的赋值。
  1. console.log(a);// a is not defined
  2. a = 1;
  3. // 当我们把var去掉的时候,结果变成了 a is not defined,很明显,没有var声明的变量,没有被提升至作用域的最前边,也就是变量提升只是对用var声明的变量而言的。
  1. 函数的内部也是如此
  1. (function(){
  2. console.log(a);//undefined
  3. var a = 1;
  4. })()
  5. 提升结果如下:
  6. (function(){
  7. var a
  8. console.log(a);//undefined
  9. a = 1;
  10. })()

2.1.2 所以暂时我们可以总结一下变量提升:

  1. 只有用var声明的变量才存在变量提升这一说法
  2. 变量提升只提升变量的声明,不会提升赋值这一部分;

3.1 函数提升

3.1.1函数的声明

函数的声明有俩种方式,一种为:

  1. function fn(){
  2. }
  3. //另一种为:
  4. var fn = function(){
  5. }
  6. // 首先的我们得知道函数属于引用类型,函数名实际上相当于一个指针,保存的是函数体所在的地址,所以函数也可以通过函数表达式var fn来声明,但是同为函数,他们的提升结果却是不同的。

3.1.2 函数式声明

  1. function fn(){
  2. console.log('hello');
  3. }
  4. fn();// hello
  5. // 结果输出为hello,这个不难理解,代码的顺序执行。函数提升后的结果还是这样。
  1. fn();// hello
  2. function fn() {
  3. console.log('hello');
  4. }
  5. // 执行结果还是hello,因为函数在解析代码的时候,同样,函数的声明被提到了作用域的最前边,如下:
  6. function fn(){
  7. console.log('hello');
  8. }
  9. fn();// hello
  10. // 需要注意的是整个function fn(){...}均为函数的定义(声明)。
  1. var fn = function(){
  2. console.log('hello');
  3. }
  4. fn();//hello
  5. // 用var声明的函数同变量一样,先把定义提升至最前边,如下:
  6. var fn;
  7. fn = function(){
  8. console.log('hello');
  9. }
  10. fn();//hello
  1. fn();//fn is not a function
  2. var fn = function(){
  3. console.log('hello');
  4. }
  5. // 但是将用var声明的函数放在后边就不行了,因为这样声明的函数,提升后是下边这样的:
  6. var fn
  7. fn();//fn is not a function
  8. fn = function(){
  9. console.log('hello');
  10. }
  11. // 同变量提升一样,提升的只是定义,并没有赋值。

4 综合运用

  1. console.log(a);
  2. function a() { //定义函数
  3. }
  4. console.log(a);
  5. var a = 3; //变量
  6. console.log(a);
  7. 运行结果如下:
  8. function a() {
  9. }
  10. function a() {
  11. }
  12. 3
  1. console.log(a);
  2. var a = 3;
  3. console.log(a);
  4. function a() {
  5. }
  6. console.log(a);

由此可见

俩次举例中,函数和变量是同名的,在代码未执行到变量赋值语句的时候,console打印出来的均为函数,变量赋值以后,打印的才是刚刚赋的值,所以由综合例子可以得出:当变量和函数同名时,函数的优先级高!

总结

由以上的例子不难看出变量提升和函数提升的特点,可以总结如下:

  1. 所有的声明都会提升到各自作用域的最顶上去。
  2. 只有用var声明的变量才存在变量提升这一说法
  3. 变量提升只提升变量的声明,不会提升赋值这一部分;
  4. 同一个变量只会声明一次,其他的会被忽略掉。
  5. 函数声明的优先级高于变量声明的优先级
  6. 所有变量的声明,在函数内部第一行代码开始执行的时候就已经完成。

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

  1. JS 函数作用域及变量提升那些事!

    虽然看了多次js函数作用域及变量提升的理论知识,但小编也是一知半解~ 这几天做了几道js小题,对这部分进行了从新的理解,还是有所收获的~ 主要参考书籍: <你不知道的JavaScript(上卷) ...

  2. JS _函数作用域及变量提升

    虽然看了多次js函数作用域及变量提升的理论知识,但也是一知半解~ 这几天做了几道js小题,对这部分进行了从新的理解,还是有所收获的~ 主要参考书籍: <你不知道的JavaScript(上卷)&g ...

  3. js 函数提升和变量提升

    总结: 函数提升比变量提升优先级高! 词法分析 词法分析方法: js运行前有一个类似编译的过程即词法分析,词法分析主要有三个步骤: 分析参数 再分析变量的声明 分析函数说明 具体步骤如下: 函数在运行 ...

  4. js函数声明提升与变量提升

    变量提升 变量提升: 在指定作用域里,从代码顺序上看是变量先使用后声明,但运行时变量的 “可访问性” 提升到当前作用域的顶部,其值为 undefined ,没有 “可用性”. alert(a); // ...

  5. 原型模式故事链(4)--JS执行上下文、变量提升、函数声明

    上一章:JS的数据类型 传送门:https://segmentfault.com/a/11... 好!话不多少,我们就开始吧.对变量提升和函数声明的理解,能让你更清楚容易的理解,为什么你的程序报错了~ ...

  6. Javascript中函数提升和变量提升

    词法分析 词法分析方法: js运行前有一个类似编译的过程即词法分析,词法分析主要有三个步骤: 分析参数 再分析变量的声明 分析函数说明 具体步骤如下: 函数在运行的瞬间,生成一个活动对象(Active ...

  7. JS中作用域和变量提升(hoisting)的深入理解

    作用域(Scoping) javascript作用域之所以迷惑,是因为它程序语法本身长的像C家族的语言.我对作用域的理解是只会对某个范围产生作用,而不会对外产生影响的封闭空间.在这样的一些空间里,外部 ...

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

    变量提升和函数提升: 就是将变量声明或者函数全部代码提升到当前作用域(全局作用域或函数作用域)最开始的部分. JavaScript中函数域为最小域范围:for循环.while循环.if语句.switc ...

  9. js笔记——js里var与变量提升

    var是否可以省略 一般情况下,是可以省略var的,但有两点值得注意: 1.var a=1 与 a=1 ,这两条语句一般情况下作用是一样的.但是前者不能用delete删除.不过,绝大多数情况下,这种差 ...

随机推荐

  1. C++多小球非对心弹性碰撞(HGE引擎)

    程序是一个月前完成的,之前一直没正儿八经的来整理下这个程序,感觉比较简单,不过即使简单的东西也要跟大家分享下. 源码下载:http://download.csdn.net/detail/y851716 ...

  2. python模块知识二 random -- 随机模块、序列化 、os模块、sys -- 系统模块

    4.random -- 随机模块 a-z:97 ~ 122 A-Z :65 ~ 90 import random #浮点数 print(random.random())#0~1,不可指定 print( ...

  3. 【iOS】Masonry 自动布局 MASViewConstraint.m:207 错误

    问题详情: Assertion failure 报错原因: make.right.equalTo([_imageView superview]).right.with.offset(-); make. ...

  4. Mybatis与Spring集成时都做了什么?

    Mybatis是java开发者非常熟悉的ORM框架,Spring集成Mybatis更是我们的日常开发姿势. 本篇主要讲Mybatis与Spring集成所做的事情,让读过本文的开发者对Mybatis和S ...

  5. PID算法通俗理解,平衡车,倒立摆,适合不理解PID算法的人来看!

    先插句广告,本人QQ522414928,不熟悉PID算法的可以一起交流学习,随时在线(PID资料再我的另一篇博客里) 倒立摆资料连接↓ https://www.cnblogs.com/LiuXinyu ...

  6. 消息中间件——RabbitMQ(一)Windows/Linux环境搭建(完整版)

    前言 最近在学习消息中间件--RabbitMQ,打算把这个学习过程记录下来.此章主要介绍环境搭建.此次主要是单机搭建(条件有限),包括在Windows.Linux环境下的搭建,以及RabbitMQ的监 ...

  7. java基础知识总结,绝对经典

    写代码: 1,明确需求.我要做什么? 2,分析思路.我要怎么做?1,2,3. 3,确定步骤.每一个思路部分用到哪些语句,方法,和对象. 4,代码实现.用具体的java语言代码把思路体现出来. 学习新技 ...

  8. Selenium模拟登陆百度贴吧

    Selenium模拟登陆百度贴吧 from selenium import webdriver from time import sleep from selenium.webdriver.commo ...

  9. 01-Spring Security框架学习--入门(二)

    一.入门案例 Spring Security 自定义登录界面 通过之前的一节 01-Spring Security框架学习--入门(一)的简单演示,Spring security 使用框架自带的登录界 ...

  10. Java 通过反射改变私有变量的值

    直接上代码 import java.lang.reflect.Field; public class Main {      public static void main(String[] args ...