Javascript执行上下文顺序的深入讲解


Posted in Javascript onNovember 04, 2020

一 执行上下文?

1什么是执行上下文?

执行上下文就是当前的 JavaScript 代码被解析和执行时所在环境的抽象概念, JavaScript 中运行任何的代码都是在执行上下文中运行的.

2 执行上下文的类型

执行上下文分为三种类型:

全局执行上下文:只有一个,这是默认的,也是基础的执行上下文.(不在任何函数中的代码都是全局执行上下文)他有两个作用,一个是创建了全局变量,也就是指向window下的变量,另一个是将this的指向全局.

函数执行上下文:有无数个,每个函数都拥有自己的执行上下文,但是只有在函数被调用的时候才会被创建,每次调用时,都会为该函数创建一个新的执行上下文…

Eval 函数执行上下文: 指的是运行在 eval 函数中的代码,很少用而且不建议使用。

二 执行上下文的生命周期

1.创建阶段

执行上下文的生命周期有三个阶段:创建阶段?执行阶段—回收阶段(主要说创建阶段)

当函数被调用,但没有执行任何其内部代码之前,会先确定这三个步骤:

1) 创建变量对象:首先初始化函数的参数 arguments,提升函数声明和变量声明
2) 创建作用域链:在执行期上下文的创建阶段,作用域链是在变量对象之后创建的.作用域链本身包含变量对象。
作用域链用于解析变量。当被要求解析变量时,JavaScript 始终从代码嵌套的最内层开始,如果最内层没有
找到变量,就会跳转到上一层父作用域中查找,直到找到该变量。
3) 确定 this 指向:有多种情况.

2. 执行阶段

执行变量赋值、代码执行

3. 回收阶段

执行上下文出栈等待虚拟机回收执行上下文

三. 变量提升和this的指向

1.变量声明提升:

大部分编程语言都是先声明变量再使用,但在 JS 中,事情有些不一样:

console.log(a); // undefined
var a = 10;

上述代码正常输出undefined而不是报错Uncaught ReferenceError: a is not defined,这是因为声明提升.
相当于:

var a; //声明 默认值是undefined “准备工作”
console.log(a);
a = 10; //赋值

2. 函数声明提升

创建函数的方法有两种,一种是通过函数声明function aa(){}

另一种是通过函数表达式var aa = function(){} ,那这两种在函数提升有什么区别呢?

console.log(f1); // function f1(){}
function f1() {} // 函数声明
console.log(f2); // undefined
var f2 = function() {}; // 函数表达式

接下来我们通过一个例子来说明这个问题:

function test() {
 aa(); // Uncaught TypeError "aa is not a function"
 bar(); // "this will run!"
 var aa = function() {
  // function expression assigned to local variable 'aa'
  alert("this won't run!");
 };
 function bar() {
  // function declaration, given the name 'bar'
  alert("this will run!");
 }
}
test();

在上面的例子中,aa()调用的时候报错了,而 bar 能够正常调用。

我们前面说过变量和函数都会上升,遇到函数表达式 var aa = function(){}时,首先会将var aa上升到函数体顶部,然而此时的 aa 的值为 undefined,所以执行aa()报错。

而对于函数bar(), 则是提升了整个函数,所以bar()才能够顺利执行。

细节必须注意:当遇到函数和变量同名且都会被提升的情况,函数声明优先级比较高,因此变量声明会被函数声明所覆盖,但是可以重新赋值。

alert(a); //输出:function a(){ alert('我是函数') }
function a() {
 alert("我是函数");
} //
var a = "我是变量";
alert(a); //输出:'我是变量'

function 声明的优先级比 var 声明高,也就意味着当两个同名变量同时被 function 和 var 声明时,function 声明会覆盖 var 声明
这代码等效于:

function a() {
 alert("我是函数");
}
var a; //hoisting
alert(a); //输出:function a(){ alert('我是函数') }
a = "我是变量"; //赋值
alert(a); //输出:'我是变量'

3. 确定this指向

// 情况1
function foo() {
 console.log(this.a) //1
}
var a = 1
foo() // this->window
// 情况2
function fn(){
 console.log(this);
}
var obj={fn:fn};
obj.fn(); //this->obj

// 情况3
function CreateJsPerson(name,age){
//this是当前类的一个实例p1
this.name=name; //=>p1.name=name
this.age=age; //=>p1.age=age
}
var p1=new CreateJsPerson("尹华芝",48);

// 情况4
function add(c, d){
 return this.a + this.b + c + d;
}
var o = {a:1, b:3};
add.call(o, 5, 7); // 1 + 3 + 5 + 7 = 16
add.apply(o, [10, 20]); // 1 + 3 + 10 + 20 = 34

// 情况5
<button id="btn1">箭头函数this</button>
<script type="text/javascript">
 let btn1 = document.getElementById('btn1');
 let obj = {
  name: 'kobe',
  age: 39,
  getName: function () {
   btn1.onclick = () => {
    console.log(this);//obj
   };
  }
 };
 obj.getName();
</script>

结果:

1 this指向window;

2谁调用了函数,谁就是 this,所以在这个场景下 foo 函数中的 this 就是 obj 对象

3 在构造函数中,this 是当前类的一个实例

4call、apply 和 bind:this 是第一个参数

5箭头函数 this 指向:箭头函数没有自己的 this,看其外层的是否有函数,如果有,外层函数的 this 就是内部箭头函数的 this,如果没有,则 this 是 window。

到此这篇关于Javascript执行上下文顺序的文章就介绍到这了,更多相关Javascript执行上下文顺序内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
用JQuery 实现的自定义对话框
Mar 24 Javascript
javascript 硬盘序列号+其它硬件信息
Dec 23 Javascript
自写的jQuery异步加载数据添加事件
May 15 Javascript
JavaScript中的lastIndexOf()方法使用详解
Jun 06 Javascript
js实现无缝滚动特效
Dec 20 Javascript
AngularJS入门教程二:在路由中传递参数的方法分析
May 27 Javascript
详解webpack异步加载业务模块
Jun 23 Javascript
React学习之事件绑定的几种方法对比
Sep 24 Javascript
浅谈Vue2.0中v-for迭代语法的变化(key、index)
Mar 06 Javascript
Angular中sweetalert弹框的基本使用教程
Jul 22 Javascript
JavaScript高级函数应用之分时函数实例分析
Aug 03 Javascript
vue使用axios上传文件(FormData)的方法
Apr 14 Javascript
解决vant中 tab栏遇到的坑 van-tabs
Nov 04 #Javascript
解决Mint-ui 框架Popup和Datetime Picker组件滚动穿透的问题
Nov 04 #Javascript
基于js实现的图片拖拽排序源码实例
Nov 04 #Javascript
在vant中使用时间选择器和popup弹出层的操作
Nov 04 #Javascript
vue 判断两个时间插件结束时间必选大于开始时间的代码
Nov 04 #Javascript
vant 时间选择器--开始时间和结束时间实例
Nov 04 #Javascript
Vue绑定用户接口实现代码示例
Nov 04 #Javascript
You might like
PHP 批量更新网页内容实现代码
2010/01/05 PHP
PHP 八种基本的数据类型小结
2011/06/01 PHP
简单实用的PHP防注入类实例
2014/12/05 PHP
php计算两个坐标(经度,纬度)之间距离的方法
2015/04/17 PHP
yii2 上传图片的示例代码
2018/11/02 PHP
javascript学习随笔(使用window和frame)的技巧
2007/03/08 Javascript
jquery异步请求实例代码
2011/06/21 Javascript
node.js中的console.timeEnd方法使用说明
2014/12/09 Javascript
jquery实现清新实用的网页菜单效果
2015/08/28 Javascript
Javascript实现单例模式
2016/01/24 Javascript
解决拦截器对ajax请求的拦截实例详解
2016/12/21 Javascript
详解如何提高 webpack 构建 Vue 项目的速度
2017/07/03 Javascript
AngularJS 中的数据源的循环输出
2017/10/12 Javascript
Vuex中的State使用介绍
2019/01/19 Javascript
JavaScript生成随机验证码代码实例
2019/09/28 Javascript
微信小程序实现锚点功能
2019/11/20 Javascript
[01:23:35]Ti4主赛事胜者组 DK vs EG 1
2014/07/19 DOTA
Python中的引用和拷贝浅析
2014/11/22 Python
python 简单照相机调用系统摄像头实现方法 pygame
2018/08/03 Python
Python基于OpenCV库Adaboost实现人脸识别功能详解
2018/08/25 Python
python单向循环链表原理与实现方法示例
2019/12/03 Python
Python递归函数特点及原理解析
2020/03/04 Python
pygame实现飞机大战
2020/03/11 Python
python实现贪吃蛇双人大战
2020/04/18 Python
python 串行执行和并行执行实例
2020/04/30 Python
amazeui页面校验功能的实现代码
2020/08/24 HTML / CSS
Bogner美国官网:滑雪服中的”Dior”
2018/01/30 全球购物
北京银河万佳Java面试题
2012/03/21 面试题
某个公司的Java笔面试题
2016/03/11 面试题
写一个函数,要求输入一个字符串和一个字符长度,对该字符串进行分隔
2015/07/30 面试题
中文师范生自荐信
2014/01/30 职场文书
三方合作协议书范本
2014/04/18 职场文书
2015年乡镇组织委员工作总结
2015/10/23 职场文书
团支部书记竞选稿
2015/11/21 职场文书
研究生毕业登记表的自我鉴定范文
2019/07/15 职场文书
spring cloud 配置中心客户端启动遇到的问题
2021/09/25 Java/Android