因为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
这样的形式并不是一个静态封装环境,静态封装环境应该是:在一个函数定义完成后立即执行,并且执行完成后返回函数中的某一个内部函数。 我们看下面一个例子,观察作用域和上下文对变量引用的影响。 复制代码 代码如下: var v = "global variable"; function method(){ writeHtml(v); writeHtml(this.v); } var Class1 = function(){ var v = "private variable"; this.v = "object variable";
由于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