this和执行上下文实现代码


Posted in Javascript onJuly 01, 2010

函数的执行上下文由当前的运行环境而定:
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 相关文章推荐
JavaScript Base64编码和解码,实现URL参数传递。
Sep 18 Javascript
常见表单重复提交问题整理及解决方法
Nov 13 Javascript
js判断是否为ie的方法小结
Jan 13 Javascript
EasyUI实现第二层弹出框的方法
Mar 01 Javascript
JavaScript实现把数字转换成中文
Jun 29 Javascript
Express实现前端后端通信上传图片之存储数据库(mysql)傻瓜式教程(一)
Dec 10 Javascript
基于JS实现bookstore静态页面的实例代码
Feb 22 Javascript
vuejs使用递归组件实现树形目录的方法
Sep 30 Javascript
原生js实现form表单序列化的方法
Aug 02 Javascript
Bootstrap Table 双击、单击行获取该行及全表内容
Aug 31 Javascript
layui2.0使用table+laypage实现真分页
Jul 27 Javascript
vue3弹出层V3Popup实例详解
Jan 04 Vue.js
jquery.validate使用攻略 第五步 正则验证
Jul 01 #Javascript
jquery validate使用攻略 第四步
Jul 01 #Javascript
jquery.validate使用攻略 第三部
Jul 01 #Javascript
jquery.validate使用攻略 第二部
Jul 01 #Javascript
jQuery Validation插件remote验证方式的Bug解决
Jul 01 #Javascript
jquery.validate使用攻略 第一部
Jul 01 #Javascript
jquery 新浪网易的评论块制作
Jul 01 #Javascript
You might like
php中对xml读取的相关函数的介绍一
2008/06/05 PHP
浅谈PHP eval()函数定义和用法
2016/06/21 PHP
PHP数组中头部和尾部添加元素的方法(array_unshift,array_push)
2017/04/10 PHP
Yii2.0使用阿里云OSS的SDK上传图片、下载、删除图片示例
2017/09/20 PHP
PHP常用日期加减计算方法实例小结
2018/07/31 PHP
php命名空间设计思想、用法与缺点分析
2019/07/17 PHP
php探针不显示内存解决方法
2019/09/17 PHP
jquery $.ajax入门应用二
2008/11/19 Javascript
浅谈关于JavaScript的语言特性分析
2013/04/11 Javascript
js读写json文件实例代码
2014/10/21 Javascript
JavaScript中的eval()函数使用介绍
2014/12/31 Javascript
浅谈js中的闭包
2015/03/16 Javascript
整理关于Bootstrap导航的慕课笔记
2017/03/29 Javascript
jQuery实现广告条滚动效果
2017/08/22 jQuery
js 公式编辑器 - 自定义匹配规则 - 带提示下拉框 - 动态获取光标像素坐标
2018/01/04 Javascript
vue1.0和vue2.0的watch监听事件写法详解
2018/09/11 Javascript
vue点击页面空白处实现保存功能
2019/11/06 Javascript
js常用方法、检查是否有特殊字符串、倒序截取字符串操作完整示例
2020/01/26 Javascript
基于vue的tab-list类目切换商品列表组件的示例代码
2020/02/14 Javascript
[50:48]LGD vs CHAOS 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/16 DOTA
[03:18]【TI9纪实】社区大触GL与木木
2019/08/25 DOTA
基于asyncio 异步协程框架实现收集B站直播弹幕
2016/09/11 Python
Python如何import文件夹下的文件(实现方法)
2017/01/24 Python
浅析使用Python操作文件
2017/07/31 Python
基于Python实现的微信好友数据分析
2018/02/26 Python
python unittest实现api自动化测试
2018/04/04 Python
python将txt文档每行内容循环插入数据库的方法
2018/12/28 Python
Python操作redis实例小结【String、Hash、List、Set等】
2019/05/16 Python
Python values()与itervalues()的用法详解
2019/11/27 Python
python GUI库图形界面开发之PyQt5动态(可拖动控件大小)布局控件QSplitter详细使用方法与实例
2020/03/06 Python
美国第一个网上卖鞋零售商:OnlineShoes.com
2017/09/24 全球购物
《大海那边》教学反思
2014/04/09 职场文书
入党介绍人评语
2014/05/06 职场文书
批评与自我批评发言稿
2014/10/15 职场文书
买卖合同协议书范本
2014/10/18 职场文书
原生Js 实现的简单无缝滚动轮播图的示例代码
2021/05/10 Javascript