`

js全局变量与局部变量 预解析与作用域链详解

    博客分类:
  • js
阅读更多
js全局变量与局部变量 预解析与作用域链详解【转】
局部变量和全局变量同名时,会隐藏这个全局变量;
4.变量

关键字: 4.变量
4.1 变量的类型
  JS是松散类型的语言

4.2 变量的声明
  var 声明是变量的标准声明
  var 声明的变量是永久性的,不能用delete运算符删除

  全局对象,调用对象初始化时,JS解析器会在相应的代码段里寻找var声明的变量,
  然后在全局对象,调用对象中创建相应的属性,此时它是未赋值的(undefined),
  当程序执行到相应的var声明代码段时才会给相应对象的属性赋值

  重复的声明:根据以上var声明作用的理解,重复的var声明不会造成任何错误,
  var声明只是方便创建全局对象,调用对象的属性,代码只是赋值用的

  遗漏的声明:给未声明的变量赋值,JS会隐式声明全局变量(在全局对象中增加相应的属性),然后给其赋值

4.3 变量的作用域
  全局变量,局部变量
  局部变量和全局变量同名时,会隐藏这个全局变量
  4.3.1 没有块级作用域
    函数中声明的变量,无论在哪里声明的,在整个函数中它们都是有定义的
  4.3.2 未声明的变量和未赋值的变量
    alert(u);会产生一个错误--使用未声明的变量会产生一个错误
    var u;alert(u);会跳出undefined---使用未赋值的变量,使用的它的默认值undefined

4.4 基本类型和引用类型
  第三章已经讲过

4.5 垃圾收集
  同java

4.6 作为属性的变量
  4.6.1 全局对象
    JS解释器开始运行时,在执行任何JS代码前,会创建一个全局对象,这个对象的属性就是JS全局变量,
    并初始化为undefined
    var声明一个JS全局变量时,实际上就是定义了一个全局对象的属性,
  
    JS解释器还会用预定义的值和函数来初始化全局对象的许多属性,如:Infinity parseInt Math
  
    非函数内部可以用this来引用这个全局对象
  
    客户端的JS中,Window对象代表浏览器窗口,它包含该窗口中的所有JS代码的全局对象,具有自我引用的window属性
  4.6.2 局部变量:调用对象
    函数的局部变量存放在调用对象的属性
    调用对象是一个完全独立的对象,所以可以防止覆盖同名的全局变量
  4.6.3 JS的执行环境
    JS解释器执行一个函数时,会创建一个执行环境
    JS允许同时存在多个全局执行环境,如:JS客户端的ifame的情况
  
4.7 深入理解变量作用域
  每个JS执行环境都有一个和它相关联的作用域链它是一个对象列表或对象链.

  查询x:变量名解析(variable name resolution)的过程,它开始查看作用域链的每一个对象,
  如果有,返回值,如果没有继续查询下一个对象,以些类推.

  作用域链的优先级:嵌套函数的调用对象>调用对象>全局对象


  根据以上理解说明JS初始化的过程:

  在JS解释器执行任何代码之前,创建全局对象
    用预定义的值和函数来初始化全局对象中的属性,eg.Math,Infinity,parseInt
    搜索函数外的var声明,创建全局对象相应的属性,初始化为undefined
  创建全局的执行环境,作用域链只有一个对象-全局对象
  依次执行代码
    遇到var声明赋值语句给全局对象相应的属性赋值
    遇到未声明赋值语句,在全局对象中增加相应的属性,并赋值
    遇到函数调用,创建调用对象
      搜索函数中的var声明和参数,创建调用对象相应的属性,初始化为undefined
      创建函数执行环境,作用域链--第一对象:调用对象;第二对象:全局对象
      依次执行代码
        遇到var声明赋值语句给调用对象相应的属性赋值
        遇到未声明赋值语句,在全局对象中增加相应的属性,并赋值
        遇到函数调用,创建嵌套函数的调用对象
          搜索嵌套函数中的var声明和参数,创建嵌套函数的调用对象相应的属性,初始化为undefined
          创建嵌套函数执行环境,作用域链--第一对象:嵌套函数的调用对象;第二对象:调用对象;第三对象:全局对象
        
    依此类推
  
    eg1.
      var scope="global";
      function f(){
        alert(scope);
        var scope="local";
        alert(scope);
      }
      f();
      过程:
      创建全局对象,搜索函数外的var声明语句,在全局对象中创建scope属性,scope=undefined
      创建全局的执行环境,作用域链只有一个对象:全局对象
      依次执行代码:
        var scope="global"时,变量名解析开始,在全局对象属性中查找scope属性
        把"global"赋给scope
        遇到函数调用:创建调用对象
          搜索函数中的var声明语句和参数,在调用对象中创建scope的属性,scope=undefined
          创建函数执行环境,作用域链:调用对象>全局对象
          依次执行代码:
            alert(scope),查询scope,变量名解析,先搜索调用对象,找到scope属性,其值为undefined,执行
            var scope="local",查询scope,变量名解析,先搜索调用对象,找到scope属性,scope="local"
            alert(scope),查询scope,变量名解析,先搜索调用对象,找到scope属性,其值为"local",执行
    
    eg2.
      var scope="global";
      function f(){
        alert(scope);
        scope="local";
        alert(scope);
      }
      f();
    过程:
      创建全局对象,搜索函数外的var声明语句,在全局对象中创建scope属性,scope=undefined
      创建全局的执行环境,作用域链只有一个对象:全局对象
      依次执行代码:
        var scope="global"时,变量名解析开始,在全局对象属性中查找scope属性
        把"global"赋给scope
        遇到函数调用:创建调用对象
          搜索函数中的var声明语句和参数,没有找到var声明语句
          创建函数执行环境,作用域链:调用对象>全局对象
          依次执行代码:
            alert(scope),查询scope,变量名解析,先搜索调用对象,没找到scope属性,再搜索全局对象,找到scope属性,其值为"global"执行
            scope="local",查询scope,变量名解析,先搜索调用对象,没找到scope属性,,再搜索全局对象,找到scope属性,scope="local"
            alert(scope),查询scope,变量名解析,先搜索调用对象,没找到scope属性,再搜索全局对象,找到scope属性,其值为"local",执行
    eg3.
      scope1="global";
      alert(scope1);
      function f(){
        alert(scope2);
        scope2="local";
      }
      f();
    过程:
      创建全局对象,没有找到var声明语句,没有自定义的全局对象属性
      创建全局的执行环境,作用域链只有一个对象:全局对象
      依次执行代码:
        scope1="global"时,变量名解析开始,作用域链是没有找到scope1属性,在全局对象属性中创建scope1属性,并赋值为"global"
        alert(scope1)时,变量名解析开始,作用域链是找到scope1属性,其值为"global",执行
        遇到函数调用:创建调用对象
          搜索函数中的var声明语句和参数,没有找到var声明语句
          创建函数执行环境,作用域链:调用对象>全局对象
          依次执行代码:

            alert(scope2),查询scope2,变量名解析,作用域链是没有找到scope2属性,报错scope2 is not defined



函数

引申一下,function. 还记得上面提到的预解析,在javascript的预解析中,除了对var 变量的预定义,还包括了提取对函数的定义,所以可以在script的任何地方定义函数,在任何地方调用。不限于它之前.

但函数的定义方式,包括了一种叫字面量定义法, 用var的方法声明function.看下面
alert(typeof y3); //结果?
var y3 = function (){ console.log('1'); }

还记得这个约定吧:调用必须出现在声明之后,为什么呢,如果理解了上面,其实这里答案已经明了。javascript 引擎在预解析 var 时 会给他们一个初始值 undefined,这样一来,如果我们在它的声明之前调用它,javascript 引擎还没拿到它的真实值,自然会报"xxx is not a function" 的错.这也理清了为什么同为函数声明,一个却关系到声明和调用的顺序,一个却无这样的约束。
结论

它是函数,是js执行,动态修改的结果,依然遵循了变量的预解析规则(在上面alert的时候,它还并没有拿到字面量函数的信息)。

如果是两个混合呢。看下面, 同时存在了为y4的变量和function。
alert(typeof y4); //结果?
function y4(){
console.log('y4')
}
var y4;

因为 javascript 在预解析时function的声明优先级高的缘故,所以y4自然为function类型, 但是在当y4 赋值之后(此时js引擎处于执行过程中),它对js的赋值操作将会覆盖function的声明。所以:
alert(typeof y5);
var y5 = 'angle';
function y5(){
console.log('ghost');
}
alert(y5);

第一次alert结果,因为它处于js 执行过程中的顶端,所以为 function。 第二次再alert时, 它的值已经被重写为5(不要被function的定义位置在下所迷惑。)

从js的解析和执行分开来想,才发现眼前豁然开朗,很多问题的答案都很自然得浮出水面,正如那篇文章作者所说,

> "一旦理解了执行环境、调用对象、闭包、词法作用域、作用域链这些概念,JS语言的很多现象都能迎刃而解。"

参考:http://blog.csdn.net/zhangw428/article/details/6060972
分享到:
评论

相关推荐

    静态全局变量,静态局部变量,全局变量,局部变量

    ### C/C++中静态全局变量、静态局部变量、全局变量及局部变量的深入解析 #### 一、概念区分 在C/C++编程语言中,**静态全局变量**、**静态局部变量**、**全局变量**及**局部变量**是经常使用的几种变量类型,它们...

    全局变量&局部变量

    ### 全局变量与局部变量深入解析 #### 一、程序内存区域概述 为了更好地理解全局变量和局部变量,我们首先需要对程序在内存中的分布有所了解。程序在运行时,操作系统会为其分配一系列的内存区域,每个区域都有...

    C语言 全局变量和局部变量详解及实例

    C语言 全局变量和局部变量详解 核心内容: 1、局部变量和全局变量 变量按照作用域分为:全局变量和局部变量 全局变量的作用域:从定义位置开始到下面整个程序结束。 局部变量的作用域:在一个函数内部定义的...

    javascript变量作用域

    JavaScript 变量作用域详解 JavaScript 变量作用域是基于其特有的作用域链的。在 JavaScript 中,变量作用域是指变量可以被访问和修改的范围。 JavaScript 没有块级作用域,而是基于函数作用域和全局作用域的。 ...

    局部变量和全局变量总汇

    ### 局部变量和全局变量知识点详解 #### 一、局部变量与全局变量的基本概念 **局部变量**指的是在函数内部定义的变量,其作用域仅限于该函数内部。这意味着,局部变量只能在其被定义的函数内部访问,并且在函数...

    全局 局部变量以及static 变量区别

    ### 全局变量、局部变量及Static变量的区别详解 #### 一、基本概念与作用域 **全局变量(Global Variables)**: - **定义**: 在函数外部定义的变量。 - **作用域**: 全局变量在整个程序中都可被访问,只要在程序的...

    js变量作用域

    ### JavaScript 变量作用域详解 #### 一、引言 在探讨JavaScript的面向对象特性之前,理解变量作用域的概念至关重要。本篇文章旨在通过一系列示例和解释,帮助读者掌握JavaScript中的变量作用域机制。 #### 二、...

    javascript中局部变量和全局变量的区别详解

    在JavaScript编程语言中,变量可以分为局部变量和全局变量两种类型,它们在作用域以及生命周期等方面有着显著的区别。 首先,全局变量的作用域是整个JavaScript代码块,包括所有函数和代码块内部。全局变量被声明在...

    Python变量作用域LEGB用法解析

    ### Python变量作用域LEGB规则详解 #### 一、引言 在编程语言中,变量的作用域是指变量可被访问的区域。对于Python这门语言来说,了解变量作用域至关重要,因为它直接影响到程序的可读性和维护性。本文将详细介绍...

    2023-04-06-项目笔记 - 第六十六阶段 - 4.4.2.64全局变量的作用域-64 -2024.03.08

    ### 全局变量的作用域详解 #### 一、引言 在编程中,变量是存储数据的基本单位。根据变量的定义位置和可见性不同,它们的作用域也有所不同。作用域决定了一个变量可以在程序中的哪些部分被访问。本篇笔记将重点...

    2023-04-06-项目笔记 - 第六十三阶段 - 4.4.2.61全局变量的作用域-61 -2024.03.05

    ### 全局变量的作用域详解 #### 一、引言 在编程中,变量是存储数据的基本单位。根据变量的定义位置和可见性不同,它们的作用域也有所不同。作用域决定了一个变量可以在程序中的哪些部分被访问。全局变量作为一种...

    2023-04-06-项目笔记 - 第六十五阶段 - 4.4.2.63全局变量的作用域-63 -2024.03.07

    ### 全局变量的作用域详解 #### 一、引言 在编程中,变量是存储数据的基本单位之一,根据其作用范围的不同,可以分为局部变量和全局变量。局部变量仅在其定义的函数或代码块内有效,而全局变量则在整个程序范围内...

    2023-04-06-项目笔记 - 第六十八阶段 - 4.4.2.66全局变量的作用域-66 -2024.03.10

    ### 全局变量的作用域详解 #### 一、全局变量的概念 在编程语言中,全局变量是指在整个程序的运行期间都可被访问的变量。这类变量通常定义在所有函数之外,因此可以在程序中的任何位置被读取或修改,除非有特定的...

    2023-04-06-项目笔记 - 第六十四阶段 - 4.4.2.62全局变量的作用域-62 -2024.03.06

    ### 全局变量的作用域详解 #### 一、引言 在编程中,变量是存储数据的基本单位。根据变量的定义位置和可见性不同,它们的作用域也有所不同。作用域决定了一个变量可以在程序中的哪些部分被访问。全局变量作为一种...

    2023-04-06-项目笔记 - 第六十二阶段 - 4.4.2.60全局变量的作用域-60 -2024.03.04

    ### 全局变量的作用域详解 #### 一、引言 在编程中,变量是存储数据的基本单位。根据变量的定义位置和可见性不同,它们的作用域也有所不同。作用域决定了一个变量可以在程序中的哪些部分被访问。全局变量作为一种...

    2023-04-06-项目笔记 - 第六十九阶段 - 4.4.2.67全局变量的作用域-67 -2024.03.11

    ### 全局变量的作用域详解 #### 一、引言 在编程中,变量是存储数据的基本单位之一,根据其作用范围的不同分为局部变量和全局变量。本篇笔记重点介绍全局变量及其作用域的相关概念,并结合C语言进行具体分析。 ##...

    2023-04-06-项目笔记 - 第六十阶段 - 4.4.2.58全局变量的作用域-58 -2024.03.02

    ### 全局变量的作用域详解 #### 一、引言 在编程中,变量是存储数据的基本单位。根据变量的定义位置和可见性不同,它们的作用域也有所不同。作用域决定了一个变量可以在程序中的哪些部分被访问。本篇笔记将重点...

    Javascript变量的作用域和作用域链详解

    JavaScript中的变量作用域和作用域链是编程中非常重要的概念,尤其对于JavaScript这种函数作用域的语言来说更是如此。本文将详细解析这两个知识点,并通过实例帮助理解。 **一、变量作用域** JavaScript中的变量...

    JS 作用域与作用域链详解

    在函数内声明的变量具有函数作用域(function scope),属于局部变量 局部变量优先级高于全局变量 代码如下: var name=”one”; function test(){  var name=”two”;  console.log(name); //two } test(); 函数...

Global site tag (gtag.js) - Google Analytics