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
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值