JS 上下文模式

更新日期: 2019-04-11阅读: 2.1k标签: 模式
function test(){
     console.log(a);//undefined;
     var a = 1;
}
test();

也许你会遇到过上面这样的面试题,你只知道它考的是变量提升,但是具体的原理又知道吗?所以我觉得很有必要搞明白底层的原理,才能加深理解,其实围绕的就是执行上下文的概念。

 

什么是执行上下文?

当控制器转到可执行的代码时,会进入该代码对应的执行上下文,可以理解为该代码对应的一个执行环境,就叫做执行上下文。  在JavaScript中运行环境有三种,分别是:

  • 全局环境:JavaScript代码执行起来,首先就是进入全局环境。
  • 函数环境:当函数被调用执行时,就会进入函数中执行。
  • eval

所以在一个JavaScript程序中,就会产生多个不同的执行上下文,这时候就需要用到前面提到的栈数据结构来管理了,我们称之为调用栈。当代码在执行过程中,遇到上面说的三种情况,就会产生三种执行上下文,然后分别压入调用栈中,等一个执行上下文执行完毕,弹出栈,才能执行下一个执行上下文中的代码,这就是栈结构的特点。

 

执行上下文的特点

  • 单线程,其实javascript就是单线程,所以很好理解。
  • 同步执行,同步就是按顺序,不能同时执行。
  • 全局上下文只有一个,它在浏览器关闭时才会弹出栈。
  • 函数的执行上下文的数目没有限制。
  • 每次某个函数被调用时,就会有新的执行上下文,即使是调用的自身函数。
function f1(){
     var n = 999;
     function f2(){
         alert(n);
    }
    return f2;
}
var result = f1();
result();//999


 我以上面这样一个例子讲解,执行上下文在调用栈中的创建过程


 

执行上下文的生命周期


如图所示,主要分为两个阶段,一个是创建阶段,一个是执行阶段。

创建阶段:
  • 生成变量对象,后面会讲解
  • 建立作用域链
  • 确定this指向
执行阶段:
  • 变量赋值
  • 函数引用
  • 执行其他代码
执行完毕后弹栈,等待回收

变量对象和活动对象的区别就在于,执行周期不一样,在创建阶段叫做变量对象,在执行阶段叫做活动对象。  

 

变量对象


变量对象的创建主要有三个阶段:

  • 1、创建arguments对象。
  • 2、检查function函数声明创建属性。在VO对象中以函数名建立一个属性,属性值为函数的地址。如果函数名的属性已经存在了,那么该属性将会被新的引用所覆盖。
  • 3、检查var变量声明创建属性。在VO对象中以变量名建立一个属性,属性值为undefined。为了防止同名的属性值会被修改为undefined,则会直接跳过,原属性值不会被修改。

举个变量提升和函数提升的例子,就明白了

function test(){
      console.log(a);
      console.log(foo());
      var a = 1;
      function foo(){
             return 2;
     }
}
test();

这是一个典型的变量提升和函数提升的例子,最后会输出undefined和2,接下来以执行上下文的生命周期来讲解

创建过程
testEC = {
      VO:{},
      scopeChain:{},
      this:{}
}
VO = {
     arguments:{},
     foo:<foo reference>,
     a:undefined
}
执行阶段
VO->AO
AO={
      arguments:{},
     foo:<foo reference>,
     a:2
}
等同于
function test(){
      function foo(){
           return 2;
     }
     var a;
     console.log(a);
     console.log(foo());
     a = 1;
}
test();

通过上面知识的讲解,进一步了解到了变量提升和函数提升的底层原理,对后面知识的学习也做了铺垫。  


链接: https://www.fly63.com/article/detial/2840

js设计模式之单例模式,javascript如何将一个对象设计成单例

单例模式是我们开发中一个非常典型的设计模式,js单例模式要保证全局只生成唯一实例,提供一个单一的访问入口,单例的对象不同于静态类,我们可以延迟单例对象的初始化,通常这种情况发生在我们需要等待加载创建单例的依赖。

前端设计模式:从js原始模式开始,去理解Js工厂模式和构造函数模式

工厂模式下的对象我们不能识别它的类型,由于typeof返回的都是object类型,不知道它是那个对象的实例。另外每次造人时都要创建一个独立的person的对象,会造成代码臃肿的情况。

JavaScript设计模式_js实现建造者模式

建造者模式:是将一个复杂的对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。工厂类模式提供的是创建单个类的模式,而建造者模式则是将各种产品集中起来进行管理,用来创建复合对象

html和xhtml,DOCTYPE和DTD,标准模式和兼容模式

主要涉及知识点: HTML与XHTML,HTML与XHTML的区别,DOCTYPE与DTD的概念,DTD的分类以及DOCTYPE的声明方式,标准模式(Standard Mode)和兼容模式(Quircks Mode),标准模式(Standard Mode)和兼容模式(Quircks Mode)的区别

前端四种设计模式_JS常见的4种模式

JavaScript中常见的四种设计模式:工厂模式、单例模式、沙箱模式、发布者订阅模式

javascript 策略模式_理解js中的策略模式

javascript 策略模式的定义是:定义一系列的算法,把它们一个个封装起来,并且使它们可以相互替换。 策略模式利用组合,委托等技术和思想,有效的避免很多if条件语句,策略模式提供了开放-封闭原则,使代码更容易理解和扩展, 策略模式中的代码可以复用。

javascript观察者模式_深入理解js中的观察者模式

javascript观察者模式又叫发布订阅模式,观察者模式的好处:js观察者模式支持简单的广播通信,自动通知所有已经订阅过的对象。存在一种动态关联,增加了灵活性。目标对象与观察者之间的抽象耦合关系能够单独扩展以及重用。

Vue中如何使用方法、计算属性或观察者

熟悉 Vue 的都知道 方法methods、计算属性computed、观察者watcher 在 Vue 中有着非常重要的作用,有些时候我们实现一个功能的时候可以使用它们中任何一个都是可以的

我最喜欢的 JavaScript 设计模式

我觉得聊一下我爱用的 JavaScript 设计模式应该很有意思。我是一步一步才定下来的,经过一段时间从各种来源吸收和适应直到达到一个能提供我所需的灵活性的模式。让我给你看看概览,然后再来看它是怎么形成的

Flutter 设计模式 - 简单工厂

在围绕设计模式的话题中,工厂这个词频繁出现,从 简单工厂 模式到 工厂方法 模式,再到 抽象工厂 模式。工厂名称含义是制造产品的工业场所,应用在面向对象中,顺理成章地成为了比较典型的创建型模式

点击更多...

内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!