JavaScript 作用域知识点梳理

JavaScript的作用域一直以来是前端开发中难以理解的知识点,对于JavaScript的作用域主要记住几句话。

一、“JavaScript” 中无块级作用域

在   Java 或 C# 中存在块级作用域,即:大括号也是一个作用域。

1 public static void main ()
2 {
3     if(1==1){
4         String name = "seven";
5     }
6     System.out.println(name);
7 }
8 // 报
View Code
public static void Main()
{
    if(1==1){
        string name = "seven";
    }
    Console.WriteLine(name);
}
// 报错
C#

在JavaScript 语言中无块级作用域

1 function Main(){
2     if(1==1){
3         var name = 'seven';
4     }
5     console.log(name);
6 }
7 // 输出: seven

补充: 标题之所以添加双影号是因为javascript6中新引入了let关键字,用于指定变量属于块级作用域。

二、JavaScript 采用函数作用域

在 JavaScript 中每个函数作为一个作用域,在外部无法访问内部作用域中的 变量。

1 function Main(){
2     var innerValue = 'seven';
3 }
4  
5 Main();
6  
7 console.log(innerValue);
8  
9 // 报错:Uncaught ReferenceError: innerValue is not defined

三、JavaScript 的作用域链

由于 JavaScript 中的每个函数 作为一个作用域,如果出现函数嵌套函数,则就会出现作用域链。

 1 xo = 'alex';
 2   
 3 function Func(){
 4     var xo = "seven";
 5     function inner(){
 6         var xo = 'alvin';
 7         console.log(xo);
 8     }
 9     inner();
10 }
11 Func();

如上述代码则出现三个作用域组成的作用域链,如果出现作用域链后,那么寻找变量的时候就会出现顺序,对于上述实例:

当执行 console.log( xo) 时,其寻找顺序 为根据作用域链从内到外的优先级寻找,如果内层没有就逐步向上找,直到没有抛出异常。

四、JavaScript 的作用域链执行前已创建,日后再去执行时只需要按照作用域链去寻找即可。

示例一

 1 xo = 'alex';
 2  
 3 function Func(){
 4     var xo = "seven";
 5     function inner(){
 6  
 7         console.log(xo);
 8     }
 9     return inner;
10 }
11  
12 var ret = Func();
13 ret();
14 // 输出结果: seven

上述代码,在函数被调用之前作用域已经存在:

   全局作用域 ->  Func函数作用域  ->  inner函数作用域

当执行【ret();】 时,由于其指代的是 inner函数,此函数作用域链在执行之前已经被定义为  全局作用域 ->  Func函数作用域  ->  inner函数作用域。 所以,在执行【ret();】时,会根据已经存在的作用域的作用域链去寻找变量。

示例二

 1 xo = 'alex';
 2  
 3 function Func(){
 4     var xo = "eirc";
 5     function inner(){
 6  
 7         console.log(xo);
 8     }
 9     xo = 'seven';
10     return inner;
11 }
12  
13 var ret = Func();
14 ret();
15 // 输出结果: seven

上述代码和示例一的目的相同,也是强调在函数被调用之前作用域链已经存在:

  • 全局作用域 -> Func函数作用域 -> inner函数作用域

不同的时,在执行【var ret = Func();】时,Func作用域中的xo变量的值已经由 “eric” 被重置为 “seven”,所以之后再执行【ret();】时,就只能找到“seven”。

示例三:

 1 xo = 'alex';<br>
 2 function Bar(){
 3     console.log(xo);
 4 }
 5  
 6 function Func(){
 7     var xo = "seven";
 8      
 9     return Bar;
10 }
11  
12 var ret = Func();
13 ret();
14 // 输出结果: alex

上述代码,在函数被执行之前已经创建了两条作用域链:

  • 全局作用域 -> Bar函数作用域
  • 全局作用域 -> Func函数作用域

当执行【ret();】时,ret代指的Bar函数,而Bar函数的作用域链已经存在:全局作用域 -> Bar函数作用域,所以,执行时会根据已经存在的作用域链去寻找。

五、声明提前

在JavaScript中如果不创建变量,直接去使用,则报错:

1 console.log(xxoo);
2 // 报错:Uncaught ReferenceError: xxoo is not defined

JavaScript中如果创建值而不赋值,则该值为 undefined,如:

1 var xxoo;
2 console.log(xxoo);
3 // 输出:undefined

在函数内如果这么写:

1 function Foo(){
2     console.log(xo);
3     var xo = 'seven';
4 }
5  
6 Foo();
7 // 输出:undefined

上述代码,不报错而是输出 undefined,其原因是:JavaScript的函数在被执行之前,会将其中的变量全部声明,而不赋值。所以,相当于上述实例中,函数在“预编译”时,已经执行了var xo;所以上述代码中输出的是undefined。

 

 

 

posted @ 2016-07-12 15:11  王志康  阅读(270)  评论(0编辑  收藏  举报