虽然看了多次js函数作用域及变量提升的理论知识,但也是一知半解~

这几天做了几道js小题,对这部分进行了从新的理解,还是有所收获的~

主要参考书籍:

《你不知道的JavaScript(上卷)》第一部分

《JavaScript高级程序设计》第四章 4.2 执行环境与作用域

理解要点:

1. js其实是一门编译语言,代码的执行分为两个阶段:(《你不知道的JavaScript(上卷)》)

a.编译阶段

b.执行阶段

2. 作用域嵌套:

在当前作用域中无法找到某个变量时,引擎就会在外层嵌套中继续查找,直到找到该变量,或是抵达最外层作用域为止。(《你不知道的JavaScript(上卷)》)

3. 包括变量和函数在内的所有声明都会在任何代码被执行前首先被处理。变量及函数声明的提升也是在这个阶段。

注意:(1)变量及函数声明的提升只是提升“声明”,并不会提升“赋值”,赋值是在函数执行阶段完成的。

   (2)函数表达式不能被提升。

4. 声明变量:使用var声明的变量会自动被添加到最近的环境中。没有使用var声明,该变量会自动被添加到全局变量中。(《JavaScript高级程序设计》)

注意:这里的没有使用var声明指的是从当前作用域向上延伸搜索,直到在全局作用域中都没有找到该变量的声明,此时该变量才会自动被添加到全局变量中。(参考上面理解要点“2”)

好啦,以下抛出题目,可以先试着做一下,你的答案是什么 ^_^

一.

 var foo = 1;
function bar( ) {
foo=10;
return function foo() { };
}
bar( );
console.log(foo);

二.

 var foo = 1;
function bar() {
foo = 10;
return;
function foo() {};
}
bar();
console.log(foo);

三.

 var a = 1;
function fn() {
console.log(a);
var a = 2;
}
fn();
console.log(a);

四.

 var a = 1;
function fn(a) {
console.log(a);
var a = 2;
}
fn(a);
console.log(a);

华丽丽的分割线———————————————————————————————————————————

哈哈,是不是so easy,以下是自己的理解,如果有误,还望大神指出^_^

一.

 var foo = 1;
function bar( ) {
foo=10;
return function foo() { };
}
bar( );
console.log(foo);

理解要点:

1. 在bar()函数作用域中,foo 变量并没有使用 var 进行声明 ,所以在执行 foo = 1; 语句时,引擎会沿着作用域链向上查找,结果在全局作用域中找到了 var foo;此时全局作用域中的 foo 的值为 1;执行了 bar() 函数中的 foo = 10;后,最终全局作用域中的变量 foo 的值为 10;

2. return function foo(){}; 在此处是一个函数表达式,而不是函数声明,所以不存在函数声明的提升。

所以:

 var foo = 1;
function bar( ) {
foo=10;
return function foo() { };
}
bar( );
console.log(foo);//

以下抛出题一的变形和答案,可以自行理解喔 ^_^

 function bar( ) {
foo=10;
return function foo() { };
}
bar( );
console.log(bar());//ƒ foo() { }
console.log(foo);//
 function bar( ) {
var foo=10;
return function foo() { };
}
bar( );
console.log(foo);//foo is not defined

二.

 var foo = 1;
function bar() {
foo = 10;
return;
function foo() {};
}
bar();
console.log(foo);

理解要点:

1. bar()函数作用域中,function foo(){};语句虽然在 return;语句后面,对的,return;语句后的所有语句都不会被执行,但是在执行代码前,函数声明会被提前处理,此时函数声明语句会被提升到当前作用域的最前面。即实际上是这样的:

 var foo = 1;
function bar() {
function foo() {};
console.log(foo);//ƒ foo() {}
foo = 10;
console.log(foo);//10
return;
function foo() {};//return;后的语句不执行
console.log(foo);
}
bar();
console.log(foo);

2. 既然 bar()函数作用域中已经有了 foo 的声明,说明 bar()中 foo 只是一个局部变量,在 bar()执行完毕就会被销毁,不影响全局作用域中的 foo 变量,所以:

 var foo = 1;
function bar() {
foo = 10;
return;
function foo() {};
}
bar();
console.log(foo);//

三.

 var a = 1;
function fn() {
console.log(a);
var a = 2;
}
fn();
console.log(a);

理解要点:

1. 函数 fn()的变量声明语句var a = 2;会提升;

2. fn ()存在函数局部变量 a;与全局作用域中的变量 a 互不影响;

所以实际上是这样的:

 var a = 1;
function fn() {
var a;
console.log(a);//undefined
a = 2;
console.log(a);//
}
fn();
console.log(a);//

即:

 var a = 1;
function fn() {
console.log(a);//undefined
var a = 2;
}
fn();
console.log(a);//

四.

 var a = 1;
function fn(a) {
console.log(a);
var a = 2;
}
fn(a);
console.log(a);

理解要点:

1. 函数形参是存在于函数作用域中的局部变量,所以此题 fn(a)中相当于声明了一个局部变量 a;

2. fn(a);相当与 fn(1);

所以:

 var a = 1;
function fn(a) {
console.log(a);
var a = 2;
}
fn(a);
console.log(a)//

又是华丽丽的分割线———————————————————————————————————————————

以下另外一些思考题,可以自行理解喔 ^_^

 var foo = 0;
(function fn1(){
foo = 1;
console.log(foo);//
(function fn2(){
var foo = 2;
console.log(foo);//
(function fn3(){
foo = 3;
console.log(foo);//
}());
}());
}());
console.log(foo);//
 (function fn1(){
(function fn2(){
(function fn3(){
foo = 3;
}());
}());
}());
console.log(foo);//
 (function fn1(){
(function fn2(){
var foo;
(function fn3(){
foo = 3;
}());
}());
}());
console.log(foo);//Uncaught ReferenceError: foo is not defined

JS _函数作用域及变量提升的更多相关文章

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

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

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

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

  3. JS基础_函数作用域

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  4. javascript中的变量作用域以及变量提升

    在javascript中, 理解变量的作用域以及变量提升是非常有必要的.这个看起来是否很简单,但其实并不是你想的那样,还要一些重要的细节你需要理解. 变量作用域 “一个变量的作用域表示这个变量存在的上 ...

  5. javascript中的变量作用域以及变量提升详细介绍

    在javascript中, 理解变量的作用域以及变量提升是非常有必要的.这个看起来是否很简单,但其实并不是你想的那样,还要一些重要的细节你需要理解变量作用域 “一个变量的作用域表示这个变量存在的上下文 ...

  6. Javascript作用域和变量提升

    下面的程序是什么结果? var foo = 1; function bar() { if (!foo) { var foo = 10; } alert(foo); } bar(); 结果是10: 那么 ...

  7. js的函数作用域跟块级作用域

    js的函数作用域跟块级作用域(原文地址:http://blog.csdn.net/huangjq36sysu/article/details/51085674)

  8. js 作用域,变量提升

    先看下面一段代码: 代码执行的结果是: 1st alert : a = 0 2nd alert : a = undefined 5th alert : a = 0 3rd alert : a = 3 ...

  9. js中变量的作用域、变量提升、链式作用域结构

    一:作用域 在ES6之前,javascript没有块级作用域(一对{}即为一个块级作用域),只有全局作用域和函数作用域(局部),因此,对应的有全局变量和局部变量.在函数内部可以访问到全局变量,但在函数 ...

随机推荐

  1. [再寄小读者之数学篇](2014-04-18 from 352558840@qq.com [南开大学 2014 年高等代数考研试题]反对称矩阵的组合)

    (2014-04-18 from 352558840@qq.com [南开大学 2014 年高等代数考研试题]反对称矩阵的组合) 设 ${\bf A},{\bf B}$ 都是反对称矩阵, 且 ${\b ...

  2. Windows系统盘符错乱导致桌面无法加载。

    问题如下 : 同事有台笔记本更换SSD硬盘,IT职员帮他将新硬盘分好区后再将系统完整Ghost过来,然后装到笔记本上.理论上直接就可以使用了!但结果开机后登陆用户桌面无法显示,屏幕黑屏什么都没有. 问 ...

  3. Python3:OOP Demo

    方便快速回顾Python的OOP语法 ###################### # 类的私有专有方法 # ###################### # __init__ : 构造函数,在生成对 ...

  4. web富文本编辑器收集

    1.UEditor 百度的. 优点:插件多,基本满足各种需求,类似贴吧中的回复界面. 缺点:不再维护,文档极少,使用并不普遍,图片只能上传到本地服务器,如果需要上传到其他服务器需要改动源码,较为难办, ...

  5. 【转】Java中文乱码的解决

    在基于Java的编程中,经常会碰到汉字的处里及显示的问题,比如一大堆乱码或问号. 这是因为JAVA中默认的编码方式是UNICODE,而中国人通常使用的文件和DB都是基于GB2312或者BIG5等编码, ...

  6. sublime自动保存设置

    首选项——用户设置 (Preferences:Settings - User) 行末添加"save_on_focus_lost": true 注意用逗号分隔 保存即可 save_o ...

  7. NOIP2016换教室

    题目 一道毒瘤概率期望DP.点这里 首先 对于每个时间段(就是每节课),我们有申请和不申请两种情况.如果不申请的话,一定在$ c[ i ] \(,否则,可能在\)c[ i ]\(,也可能在\)d[ i ...

  8. Java 模板模式

    定义:定义了一个算法的骨架,并允许子类为一个或多个步骤提供实现 模板方法使得子类可以在不改变算法的结构的情况下,重新定义算法的某些步骤 类型:行为型 一次性实现一个算法的不变的部分,并将可变的行为留给 ...

  9. python---二叉树遍历

    重学. # coding = utf-8 # 二叉树遍历 class Node: """节点类""" def __init__(self, ...

  10. python中的while循环和for循环

    1.while循环 Gif 演示 Python while 语句执行过程 while 语句时还有另外两个重要的命令 continue,break 来跳过循环,continue 用于跳过该次循环,bre ...