Javascript中this关键字通常指向当前函数的拥有者。在javascript中通常把这个拥有者叫做执行上下文。函数的执行上下文由当前的运行环境而定:
1. 全局变量和全局函数附属于全局对象(window),因此使用”var”或”this”两种方法定义全局变量是等效的。
2. 执行上下文和作用域不同。执行上下文在运行时确定,随时可能改变,而作用域则在定义时确定,永远不会变。
3. 如果当前执行的是一个对象的方法,则执行上下文就是这个方法所附属的对象。
4. 如果当前是一个创建对象的过程或者执行一个对象的方法,则执行上下文就是这个正在被创建的对象。
5. 如果一个方法在执行时没有明确指定附属对象,则这个方法的上下文为全局对象。
6. 使用call和apply可以改变对象的执行上下文。
看下面的例子:
var v1 = "global variable"; //全局变量附属于对象
//this.v1 = "global variable with this"; //全局变量定义时使用var v1和this.v1两种方法等效。
function func1(){
var v1 = "part variable";
writeHtml(v1);
writeHtml(this.v1);
}
func1(); //part variable
//global variable
因为func1中有和全局对象同名的v1变量,所以在func1中直接引用v1引用的是func1中定义的变量。javascript同样有局部变量隐藏全局变量的特性。但func1没有明确的指定附属对象,因此他的执行上下文是全局对象,使用this引用变量的是全局变量。
再看一个稍微复杂一点的例子:
function ftest(){
var v = "v1v1v1";
this.this_v = "this_v";
return function(){
writeHtml(v);
writeHtml(this.this_v);
}
}
var a = ftest();
var v = "v2v2v2";
writeHtml(this_v); // this_v
a(); //v1v1v1
//this_v
当ftest当做函数来执行时,上下文为全局对象。所以在ftest中使用this定义的变量成为了全局变量。所以我们在ftest外面直接使用变量名访问this_v的值。但是,由于ftest中返回的匿名函数是定义在ftest内部的,所以这个匿名函数的作用域就是在ftest内部。因此当有全局变量v和局部变量v同名时,这个匿名函数访问到的是ftest内部定义的变量v。
接下来把ftest当做类,使用new关键字来实例化:
function ftest(){
var v = "v1v1v1";
this.this_v = "this_v";
return function(){
writeHtml(v);
writeHtml(this.this_v);
}
}
var a = new ftest();
var v = "v2v2v2";
//writeHtml(this_v); // 错误:this_v未定义
a(); //v1v1v1
//undefined
把ftest当做对象来实例化时,在对象的创建过程中,上下文为被创建的对象本身。注意,这个时候创建的对象是ftest的实例,而创建完成以后又返回了一个函数,这导致了new ftest()实例化后返回的是一个函数,而不是ftest()实例化后对象的引用。因此,这个已经实例化的对象无法被引用。当我们定义这个被返回的函数时,因为没有用this指定这个函数的上下文,因此这个被返回的函数上下文为全局对象,作用域为ftest()函数内部。所以函数a()执行时的由于上下文中没有定义this_v变量,导致了访问错误。
注意,上面的代码:
function ftest(){
return function(){
}
}
这样的形式并不是一个静态封装环境,静态封装环境应该是:在一个函数定义完成后立即执行,并且执行完成后返回函数中的某一个内部函数。
我们看下面一个例子,观察作用域和上下文对变量引用的影响。
var v = "global variable";
function method(){
writeHtml(v);
writeHtml(this.v);
}
var Class1 = function(){
var v = "private variable";
this.v = "object variable";
var method2 = method;
this.method2 = method;
var method3 = function(){
writeHtml(v);
writeHtml(this.v);
}
this.method3 = function(){
writeHtml(v);
writeHtml(this.v);
}
method2(); //global variable
//global variable
this.method2(); //global variable
//object variable
method3(); //private variable
//global variable
this.method3();//private variable
//object variable
}
var obj = new Class1();
由于method在全局中定义,所以method的作用域在定义的时候就被确定为全局的。所以method2在Class1内部被调用时,其作用域与是全局,上下文是全局对象。因此,在函数中访问到的变量都是全局变量。
同理,this.method2在被调用时,其作用域是全局,但是由于该函数在定义时使用this关键字指明了其上下文为Class1的对象,所以在该函数访问没有上下文限定的变量时访问到的是全局变量,访问有上下文限定的变量时为访问到的是当前上下文中对应的变量。
在调用method3和this.method3时,在访问没有上下文限定的变量时访问到的是局部变量,因为局部变量隐藏了全局变量。有上下文限定时和method2相同,访问到的是当前上下问文中的变量。
使用call和apply可以改变执行上下文,由于call和apply只是参数类型不一样,因此例子下面都用call来演示。
var v = "global variable";
var method = function(){
writeHtml(this.v);
}
var Class2 = function(){
this.v = "object variable in instance of Class2";
this.method = function(){
writeHtml(this.v);
}
}
var Class3 = function(){
this.v = "object variable in instance of Class3";
this.method = function(){
writeHtml(this.v);
}
}
var obj2 = new Class2();
var obj3 = new Class3();
method(); //global variable
obj2.method(); //object variable in instance of Class2
obj3.method(); //object variable in instance of Class3
method.call(obj2); //object variable in instance of Class2
method.call(obj3); //object variable in instance of Class3
obj2.method.call(obj3); //object variable in instance of Class3
obj2.method.call(this); //global variable
obj3.method.call(obj2); //object variable in instance of Class2
obj3.method.call(this); //global variable
可以看到,使用call或apply可以将方法绑定到指定的上下文中。在全局环境中this指向的上下文为全局对象。
分享到:
相关推荐
执行上下文可以理解为当前 JavaScript 代码被评估和执行时的一个环境,其中包含了变量、函数声明、作用域链以及 `this` 的值。函数堆栈则描述了函数调用时的顺序和结构,而变量提升则是 JavaScript 中的一个特殊机制...
JavaScript中的执行上下文和调用堆栈是理解JS运行机制的关键概念。执行上下文是代码在特定环境中运行的抽象表示,它定义了变量、函数以及`this`关键字的行为。执行上下文主要有三种类型:全局执行上下文(默认的运行...
JavaScript 是一种解释型的、基于原型的编程语言,其运行机制是通过执行上下文和执行栈来管理代码的执行过程。执行上下文是JavaScript引擎在处理代码时的一个关键概念,它定义了代码执行的环境,包括变量、函数、...
JavaScript中的`this`和执行上下文是理解JavaScript行为的关键概念,它们决定了代码中变量和方法的查找路径。本文将深入探讨这两个概念。 首先,`this`关键字在JavaScript中表示当前执行环境的对象。它取决于函数...
当使用`new`关键字调用函数时,会创建一个新的执行上下文,`this`会指向这个新实例。 ```javascript function Person(name) { this.name = name; } var person = new Person("Alice"); console.log(person....
执行上下文就是当前的 JavaScript 代码被解析和执行时所在环境的抽象概念, JavaScript 中运行任何的代码都是在执行上下文中运行的. 2 执行上下文的类型 执行上下文分为三种类型: 全局执行上下文:只有一个,这是默认...
它分为全局执行上下文、函数执行上下文和`eval`执行上下文。全局上下文是默认的代码运行环境,当代码加载时,JavaScript引擎首先进入的就是全局上下文。函数执行上下文在调用函数时创建,而`eval`执行上下文则在`...
执行上下文分为全局执行上下文、函数执行上下文和eval代码执行上下文三种主要类型。 1. 全局执行上下文(Global Context): - 全局执行上下文是执行环境的起始点,也是唯一一个在程序开始时就存在的执行上下文。...
在JavaScript中,事件循环和执行上下文是理解代码运行过程的关键概念。当我们浏览器输入URL并按下回车时,一系列复杂的操作开始进行。首先,浏览器获取HTML代码,并创建一个栈内存来为执行上下文提供存储环境。栈是...
JavaScript中的执行上下文和堆栈是理解其工作原理的关键概念,尤其对于JavaScript的异步处理和作用域至关重要。首先,执行上下文(Execution Context)是JavaScript引擎用来存储变量、函数和作用域规则的地方,它是...
JavaScript中,作用域和上下文是理解...总之,JavaScript的作用域决定了变量的生命周期和可见性,而上下文则关乎`this`关键字的值,指示了代码执行的环境。理解这两者对于编写可维护和高效的JavaScript代码至关重要。
在学习JavaScript的过程中,理解和掌握执行栈与执行上下文是相当重要的,因为它们是理解JS运行机制的核心。执行栈是JavaScript引擎用来追踪函数调用的一种数据结构,而执行上下文则是JS代码执行的环境。下面将详细...
JS执行上下文是理解JavaScript运行机制的关键概念,它决定了代码如何在内存中运行和作用域链的形成。本文将通过实例深入解析JS执行上下文的运行原理。 首先,我们注意到JS代码的执行顺序并不总是按照代码的书写顺序...
执行上下文是JavaScript语言中一个非常核心的概念,它是当前JavaScript代码被解析和执行时所在环境的抽象概念。理解执行上下文对深入掌握JavaScript的工作机制至关重要。 JavaScript代码运行在三种基本的执行上下...
执行上下文主要分为全局执行上下文、函数执行上下文和eval执行上下文三种类型。 全局执行上下文是当JavaScript代码首次运行时自动进入的环境,它是默认的执行上下文。在这个上下文中,可以访问全局对象(在浏览器中...
本文实例讲述了javascript执行上下文、变量对象。分享给大家供大家参考,具体如下: 突然看到一篇远在2010年的老文,作者以章节的形式向我们介绍了ECMA-262-3的部分内容,主要涉及到执行上下文、变量对象、作用域、...