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的一些看法
May 27 Javascript
js中使用DOM复制(克隆)指定节点名数据到新的XML文件中的代码
Jul 27 Javascript
JS左右无缝滚动(一般方法+面向对象方法)
Aug 17 Javascript
jQuery拖动div、移动div、弹出层实现原理及示例
Apr 08 Javascript
Angular的$http与$location
Dec 26 Javascript
利用C/C++编写node.js原生模块的方法教程
Jul 07 Javascript
js判断输入框不能为空格或null值的实现方法
Mar 02 Javascript
vue实现提示保存后退出的方法
Mar 15 Javascript
微信小程序分享功能之按钮button 边框隐藏和点击隐藏
Jun 14 Javascript
vue实现瀑布流组件滑动加载更多
Mar 10 Javascript
微信小程序自定义顶部组件customHeader的示例代码
Jun 03 Javascript
解决VUE自定义拖拽指令时 onmouseup 与 click事件冲突问题
Jul 24 Javascript
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+mysql+ajax轻量级聊天室实现方法详解
2016/10/17 PHP
thinkphp5.1 文件引入路径问题及注意事项
2018/06/13 PHP
PHP递归统计系统中代码行数
2019/09/19 PHP
jquery cookie插件代码类
2009/05/26 Javascript
浅析js中2个等号与3个等号的区别
2013/08/06 Javascript
js写出遮罩层登陆框和对联广告并自动跟随滚动条滚动
2014/04/29 Javascript
Javascript定义类(class)的三种方法详解
2015/03/13 Javascript
浅析在javascript中创建对象的各种模式
2016/05/06 Javascript
Node.js的环境安装配置(使用nvm方式)
2016/10/11 Javascript
vue2.0实战之基础入门(1)
2017/03/27 Javascript
深入理解jquery的$.extend()、$.fn和$.fn.extend()
2017/07/08 jQuery
Vue.js 通过jQuery ajax获取数据实现更新后重新渲染页面的方法
2018/08/09 jQuery
详解vantUI框架在vue项目中的应用踩坑
2018/12/06 Javascript
javascript实现的字符串转换成数组操作示例
2019/06/13 Javascript
如何使用Jquery动态生成二级选项列表
2020/02/06 jQuery
[57:53]Secret vs Pain 2018国际邀请赛小组赛BO2 第二场 8.17
2018/08/20 DOTA
Python中的类与对象之描述符详解
2015/03/27 Python
对Python新手编程过程中如何规避一些常见问题的建议
2015/04/01 Python
python获取网页中所有图片并筛选指定分辨率的方法
2018/03/31 Python
flask中的wtforms使用方法
2018/07/21 Python
详解如何为eclipse安装合适版本的python插件pydev
2018/11/04 Python
python实现转盘效果 python实现轮盘抽奖游戏
2019/01/22 Python
django 2.2和mysql使用的常见问题
2019/07/18 Python
Python 仅获取响应头, 不获取实体的实例
2019/08/21 Python
Python中函数的返回值示例浅析
2019/08/28 Python
Python decimal模块使用方法详解
2020/06/08 Python
HTML中fieldset标签概述及使用方法
2013/02/01 HTML / CSS
Europcar德国:全球汽车租赁领域的领导者
2018/08/15 全球购物
在线购买澳大利亚设计师手拿包和奢华晚装手袋:Olga Berg
2019/03/20 全球购物
戴森英国官网:Dyson英国
2019/05/07 全球购物
抽象方法、抽象类怎样声明
2014/10/25 面试题
公务员试用期满考核材料
2014/05/22 职场文书
2015年元宵节活动总结
2015/02/06 职场文书
2019年大学生暑期社会实践调查报告模板
2019/11/07 职场文书
Nginx配置80端口访问8080及项目名地址方法解析
2021/03/31 Servers
Golang数据类型和相互转换
2022/04/12 Golang