JS 作用域以及变量提升

变量作用域又叫变量的可见性

在 JavaScript 中作用域可以分为两种:

  1. 全局作用域
  2. 块级作用域 (局部作用域)

全局作用域

全局作用域比较简单,如果一个变量的作用域是全局的,那么这个变量可以在全局的任何一个地方使用。

在全局作用域中起作用的变量就叫做全局变量。

var a = 0;
function test() {
  return a++;
}
test();
console.log(a);
function test() {
  var a = 123;
  b = 123;
}
test();
console.log("a:", a);
console.log("b:", b);

块级作用域

一个变量在块级区域内起作用,那么这个变量的作用域就是块级作用域。

function test() {
  var a = 0;
  return a + 1;
}
test();
console.log(a);

全局变量实现了一处声明,到处使用。

虽然全局变量使用起来比较方便,但是尽量不使用全局变量。(为什么?)

var a = 1;
function test() {
  var a = 5;
  console.log(a);
}
console.log(a);
test();

作用域链

在 JavaScript 中,一切皆对象。函数也和其他对象一样,拥有可以通过代码访问的属性和一系列仅供 JavaScript 引擎访问的内部属性。其中的一个内部属性就是[[Scope]],该内部属性包含了函数被创建的作用域中的对象集合,这个集合也就被称为作用域链。

我们在执行一个函数时,会创建一个称为“执行期上下文”的内部对象,执行期上下文定义了函数执行时的环境。每一个执行期上下文都有自己的作用域链,用于标识符解析,当执行期上下文被创建的时候,它的作用域初始化为当前运行函数的[[Scope]]所包含的对象。

这些值按照它们出现在函数的顺序被复制到执行期上下文的作用域链中。它们共同组成了一个新的对象,叫做“活动对象(activation object)”,该对象包含了函数的所有局部变量、命名参数、参数集合以及 this,然后此对象会被推入作用域的前端,当执行期上下文被销毁,活动对象也随之销毁。

我们可以将全局作用域的活动对象称为 GO,局部作用域的活动对象称为 AO。

var a = 100;
function test() {
  var b = 200;
  var c = 300;
  console.log(a);
  console.log(b);
  console.log(c);
}

将上面代码的作用域以图表的形式展示出来:

1.png

变量提升

我们可以通过几行代码来看一下变量提升表现出来的情形:

// 1.
console.log(test);
var test = 100;
// 2.
console.log(test);
function test() {}

那么为什么会出现这种情况呢?

我们首先得了解 js 是怎么运行的?

  1. 语法解析
    通篇扫描代码,查看代码是否存在语法上的错误,语法解析并不执行代码
  2. 预编译
  3. 解释执行
    解释一行,执行一行
    预编译的过程,就是出现这种情况的原因。

预编译过程

预编译发生在函数执行的前一刻。

我们在使用函数或者变量之前都会去声明这个函数或变量,声明后我们才能够使用。

  1. 未经声明就赋值的变量就是全局变量
  2. 一切声明的全局变量全是 window 的属性
// 变量声明
var a = 100;
// 函数声明
function test() {
  // 使用变量
  console.log(a);
}
// 使用函数
test();

在预编译过程中,会将所有的声明提到函数体的最前面,也就是声明提升。声明提升有两条规则:

  1. 函数声明 整体提升
  2. 变量声明 声明提升
// 例子:
var a = 100;
function test() {
  a = 200;
  var b = 10;
  function fn() {}
  b = 300;
}
test();
// 变量提升后:
var a;
a = 100;
function test() {
  // 预编译
  var b;
  function fn() {}
  // 预编译结束
  // 函数执行
  a = 200;
  b = 10;
  b = 300;
  // 函数执行完成
}
test();

这就是声明提升的最基础的体现。

小测试:

// 例子 1:
var a = 100;
function test() {
  console.log("a:", a);
  console.log("b:", b);
  a = 200;
  var b = 10;
  console.log("b:", b);
  function fn() {}
  b = 300;
  c = 200;
}
test();
console.log("c:", c);
// 例子 2:
var a = 100;
function test() {
  console.log("a:", a);
  console.log("b:", b);
  var a = 200;
  console.log("a:", a);
  var b = 10;
  console.log("b:", b);
  function b() {}
  b = 300;
  c = 200;
}
test();
console.log("c:", c);

声明提升的过程中,有以下 4 个步骤:

  1. 创建 执行期上下文 AO (Activation Object)对象
  2. 找形参和变量声明,将变量和形参名作为 AO 属性名,值为 undefined
  3. 将实参值和形参值统一
  4. 在函数体里面找函数声明,值赋予函数体
// 测试 1:
function fn(a) {
  console.log("a 0:", a);
  var a = 123;
  console.log("a 1:", a);
  function a() {}
  console.log("a 2:", a);
  var b = function() {};
  console.log("b 0:", b);
  function d() {}
  console.log("d 0:", d);
}
fn(1);
// 测试 2:
function test(a, b) {
  console.log(a);
  c = 0;
  var c;
  a = 3;
  b = 2;
  console.log(b);
  function b() {}
  function d() {}
  console.log(b);
}
test(1);
// 测试 3:
function test(a, b) {
  console.log(a);
  console.log(b);
  var b = 234;
  a = 123;
  function a() {}
  var a;
  console.log(a);
  b = 234;
  console.log(b);
  var b = function() {};
  console.log(b);
}
test(1);
// 测试 4:
function bar() {
  return foo;
  foo = 10;
  function foo() {}
  var foo = 11;
}
// 测试 5:
function bar() {
  foo = 10;
  function foo() {}
  var foo = 11;
  return foo;
}
// 测试 6:
a = 100;
function demo(e) {
  function e() {}
  arguments[0] = 2;
  console.log(e);
  if (a) {
    var b = 123;
    function c() {}
  }
  var c;
  a = 10;
  var a;
  console.log(b);
  f = 123;
  console.log(c);
  console.log(a);
}
var a;
demo(1);
console.log(a);
console.log(f);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343