JavaScript的作用域和变量对象

时间:2022-09-02 11:20:25

变量对象

先来说说什么是变量对象。变量对象中又存储了什么东西吧。

JavaScript中的运行环境包含全局运行环境和函数运行环境这两种,每进入到一个运行环境都会创建一个变量对象,这个对象中记录了在当前运行环境中能够訪问到的变量,它们以变量对象的属性形式存在。也就是说这个变量对象成为“作用域”这个抽象概念的实体。

同一时候,变量对象中的属性记录是有一定先后顺序的。而且属性值究竟是实际的值还是undefined也是分阶段的(进入上下文(函数開始调用,但还未运行内部的详细代码)。运行代码阶段)。

1. 在全局环境中:

(1)函数声明

(2)变量声明—其值为undefined,一直到运行到这条语句的时候才被赋予详细值。

2. 函数环境中:

在进入运行环境的时候,变量对象会进行例如以下初始化:

(1)arguments对象,对象中的值被赋予详细的实參值

(2)函数的形參:变量对象的一个属性。其属性名就是形參的名字。其 值就是实參的值;对于没有传递的參数,其值为undefined。

(3)函数声明:变量对象的一个属性。其属性名和值都是函数对象创建出来的,其值为指向某个函数对象的引用;

(4)变量声明:变量对象的一个属性,其属性名即为变量名。其值为undefined。

而在运行到函数内部的详细某个语句的时候。上面所述的值为undefined的变量。其值都会被赋予详细的值。

总结一句话:在进入上下文的时候(比方进入到一个全局环境或者是调用某个函数),变量对象除了arguments。函数的声明,以及參数被赋予了详细的属性值,其他的变量属性默认的都是undefined。

关于声明提升

**

JavaScript具有“声明提升”的特性。使用function或var进行的声明会被提升至当前作用域的顶端。

**

事实上理解这句话能够从进入到一个运行环境后都发生了什么来看,正如上面所说。进入到一个上下文后,声明的函数(值为函数对象的引用)和变量(其值为undefined)都被记录在了变量对象中。到了运行代码阶段,一边在作用域链中搜索标识符,一边运行到某个语句后把原本进入上下文阶段值为undefined的变量更换为详细的值。

函数声明提升

看看第(3)条,在进入到运行环境的时候。函数声明就会被加入到变量对象中,其相应的属性值是指向所声明的函数对象的引用,假设在函数声明的语句之前存在一条函数调用语句,引擎会在作用域中搜索该标示符。而且在变量对象中成功查找到了声明的函数名,其值也是存在的,即指向创建的函数对象的引用。因此这样写代码不会报错,成功调用了函数。

而通过函数表达式方式,也就是说不是以function开头,而是把函数赋值给了某个变量,由于在进入上下文阶段是不会给变量赋予详细的值的(參看第(4)条),所以这个时候该变量的值为undefined,要等到运行代码阶段,运行到这条语句后才会将其值更换为指向函数对象的引用,因此假设调用函数的语句在前面则会报错。此时在变量对象中搜索标识符。其值还是undefined,当然不能成功调用了。

再来看一个变量声明提升的样例

var a = "Hello";
function b() {
alert(a); //undefined
var a = "World";
alert(a); //World
}
b();

为什么运行了b(),第一个alert(a)是undefined呢?来看看进入到全局运行环境中发生了什么。全局环境的变量对象被填充为:

VO={

       b:指向一个函数对象的引用,

       a:undefined—>运行到该句话后变成”hello”

}

再来看看调用b()发生了什么,其变量对象被填充为:

VO={

        arguments:{},

        a:undefined—->运行到详细语句后才被赋值为“world”

}

注意到第一次运行到 alert(a)时会在当前作用域中搜索标识符a,发现已经存在一个a(则不会再沿着作用域链往外层搜索了),其值是undefined。而运行到var a = “World”;时则被替换为了详细的值。

假设在b中去掉var。则会沿着作用域链搜索到全局环境中的变量a。那么两个alert(a)都会输出“hello”。

參考资料:javascript 运行环境。变量对象,作用域链

JavaScript中的作用域与变量声明提升

JavaScript的作用域和变量对象的更多相关文章

  1. javascript执行上下文和变量对象

    执行上下文(execution context): 执行上下文就是当前 JavaScript 代码被解析和执行时所在环境的抽象概念. js语言是一段一段的顺序执行,这个“段”其实就是我们说的这个执行上 ...

  2. javascript 执行环境,变量对象,作用域链

    前言 这几天在看<javascript高级程序设计>,看到执行环境和作用域链的时候,就有些模糊了.书中还是讲的不够具体. 通过上网查资料,特来总结,以备回顾和修正. 要讲的依次为: EC( ...

  3. JavaScript——执行环境、变量对象、作用域链

    前言 这几天在看<javascript高级程序设计>,看到执行环境和作用域链的时候,就有些模糊了.书中还是讲的不够具体.通过上网查资料,特来总结,以备回顾和修正. 目录: EC(执行环境或 ...

  4. 【转】javascript 执行环境,变量对象,作用域链

    这篇文章比较清晰的解释了一些作用域链相关的概念,忍不住收藏了 原文地址:http://segmentfault.com/a/1190000000533094 前言 这几天在看<javascrip ...

  5. 再看javascript执行上下文、变量对象

    突然看到一篇远在2010年的老文,作者以章节的形式向我们介绍了ECMA-262-3的部分内容,主要涉及到执行上下文.变量对象.作用域.this等语言细节.内容短小而精悍,文风直白而严谨,读完有酣畅淋漓 ...

  6. JavaScript词法作用域与调用对象

    关于 Javascript 的函数作用域.调用对象和闭包之间的关系很微妙,关于它们的文章已经有很多,但不知道为什么很多新手都难以理解.我就尝试用比较通俗的语言来表达我自己的理解吧. 作用域 Scope ...

  7. JavaScript 执行环境 与 变量对象

    什么是JS的执行环境? function funA(){ //一段代码静静的躺在这里,不能叫执行环境 } funA(); //当代码开始执行以后,系统会将它存入执行栈,并为他准备好足够的内存空间使用 ...

  8. js中函数的创建和调用都发生了什么?执行环境&comma;函数作用域链&comma;变量对象

    博客搬迁,给你带来的不便,敬请谅解! http://www.suanliutudousi.com/2017/11/26/js%E4%B8%AD%E5%87%BD%E6%95%B0%E7%9A%84%E ...

  9. JavaScript的作用域和闭包

    首发于:https://mingjiezhang.github.io/ 闭包和作用域有着千丝万缕的联系. js的作用域 具体的作用域我就不展开叙述了.其中很重要的两点就是:js的作用域链机制和函数词法 ...

随机推荐

  1. 时间&OpenCurlyDoubleQuote;Thu Aug 14 2014 14&colon;28&colon;06 GMT&plus;0800”的转换

    var date = "Thu Aug 14 2014 14:28:06 GMT+0800"; var va = DateTime.ParseExact(date, "d ...

  2. rspec&plus;rest-client测试第三方web service

    如果你手工测试Restful 服务将会是一件非常单调乏味的事情.当然,目前有一些浏览器插件可以通过可视化的界面帮助你手工测试,例如postman.rest console,但是每次系统版本更新,你都需 ...

  3. 大数据及hadoop相关知识介绍

    一.大数据的基本概念 1.1什么是大数据 互联网企业是最早收集大数据的行业,最典型的代表就是Google和百度,这两个公司是做搜索引擎的,数量都非常庞大,每天都要去把互联网上的各种各样的网页信息抓取下 ...

  4. springboot~Integer和int如何选择,Integer的意义何在

    今天说一下自己在项目中遇到的问题,然后总结一下Integer引用类型和int值类型 关于默认值 Integer默认为null int默认为0 为什么把数据实体设计成Integer或者不是int 大叔认 ...

  5. MySQL的随机数函数rand&lpar;&rpar;的使用技巧

    咱们学php的都知道,随机函数rand或mt_rand,可以传入一个参数,产生0到参数之间的随机整数,也可以传入两个参数,产生这两个参数之间的随机整数. 而在mysql里,随机数函数rand不能传参, ...

  6. iOS12 XCode10更新

    原因:libc++.tbd库取代了libstdc++.6.0.9.tbd库 解决方法:我在项目里去掉了libstdc++.6.0.9.tbd库 这个时候去编译还是会报错, 解决方法:Xcode-fil ...

  7. 关于i&plus;&plus;和i&plus;&plus;的左值、右值问题

    1.什么是左值和右值? 左值就是出现在表达式左边的值(等号左边),可以被改变,他是存储数据值的那块内存的地址,也称为变量的地址: 右值是指存储在某内存地址中的数据,也称为变量的数据. 左值可以作为右值 ...

  8. CKEditor 5

    1.官网 https://ckeditor.com/ckeditor-5/download/ 2.

  9. LR监控Apache资源

    前提本文使用的是lampp环境下自带的Apache服务 步骤1.修改Apache中Httpd.conf文件,添加如下代码:文件位置为:/opt/lampp/etc/httpd.conf,如下图: &l ...

  10. HAL库PWM

    1.占空比控制 a.在CUBMX的TIM下选择时钟源,选择通道为模式PWM Generation ch1 ,设置分频系数,初始值,不需要自动重装载,选择PWM模式1或2,设置比较值pulse,设置初始 ...