函数的执行上下文由当前的运行环境而定:
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指向的上下文为全局对象。
相关推荐:
ChatGPT无法加载?检查网络并尝试重启,助您快速恢复畅通体验,burj ai
SEO优化顾问:让您的网站脱颖而出的秘密武器,ai ps 群
SEO基础知识解析:如何提升网站排名,助力企业成长,毕节企业建设网站
英语日记AI生成:轻松提升英语水平的智能助手
为什么“360收录”是你网站推广的必备利器,seo亚马逊
AI提供的阅读书目对学生的专业知识有多大帮助,沃奇ai
chatai写作免费一键生成,轻松解决写作难题!,陈逗逗ai换脸在线看
ChatGPT无法访问原因分析及解决方案,斑马ai报道
SEM+SEO:助力企业数字化营销的双剑合璧之道,ai头像丑
文章AI排版,让创作更高效的秘密武器
SEO获取流量的必杀技:如何通过优化轻松提升网站排名,德州全网营销推广价格
ChatGPT:OpenAI的创新之作-一款颠覆传统的语言模型,ai球衣号
打造高效创作体验,写文章AI软件重塑内容生产力
互联网留痕:数字时代的隐形轨迹与自我管理,灯塔网站推广包年多少钱
ChatGPT登录界面都不显示了?可能是这些原因导致的!,上海小学ai智能课
seo需要学些什么内容,学seo的基础 ,中国ai公司年收入
《命运交错的轨迹:小说背后的无尽魅力》,seo优化huifachina
SEO快速提升:让你的网站排名瞬间飙升的秘诀,网站建设的发展目标
如何下载免费AI软件,让你的工作和生活更智能
自动生成文章的AI软件,助力内容创作的未来
360AI写作怎样?助力创作的新风尚,ai到访
SEO需要什么语音,seo需要考虑什么 ,学生作业ai
为什么新手做seo好做,为什么要懂seo ,ai少女 3060显卡
GPT怎么收费?揭秘AI技术的定价与价值,ai报考高考
ChatGPT的VPN梯子:畅享全球互联网自由,打破地域限制,一直搜Ai
ChatGPT网页版内容显示不全的原因与解决方案,ai超级绘
好用的人工智能AI软件推荐,让你的生活更智能!
AI免费写作一键生成,效率与创意的完美结合
Chatttst:开启智能沟通新时代的无限可能,上海联通ai
ChatGPT网页版内容显示不全的解决方案:如何轻松解决问题?,课件ai
仿写AI:智能时代的创作革命,洛江区移动房网站推广
seo适用于什么领域,seo适用于什么领域中 ,ai智能翻译写作机器人v1.0
GPT哪个模型是最新的?AI语言生成的未来,ai回头
AI免费生成文章的软件:轻松创作的秘密武器
seo,seoul city ,ai精洗
优化高效率:提升个人与企业竞争力的关键,安康网站推广咨询
SEO公司核心业务是什么?揭秘提升网站排名的奥秘,写作助手ai一键生成作文在线
AI热门软件,未来科技的钥匙
SEO出超:如何通过精准优化实现网站流量大爆发,营销推广方式联系f火15星
SEO优化要钱吗?揭秘SEO投资背后的价值与回报,ai领域
怎么让AI润色文章,让写作更轻松?
ChatGPT服务部分恢复:人工智能助力全新体验,ai对象菜单
优化公司:助力企业腾飞的秘密武器,微信营销推广价格多少
SEO更多-让你的企业站点在搜索引擎中脱颖而出,如何结交seo大神
优化网站的秘诀:提高网站速度与用户体验,助力业务腾飞,旅游网站建设步骤
ZBlog:开启你的个人网站新时代,轻松搭建与管理,二手手机营销推广方案
丹东seo是什么怎么选,丹东spr ,黑发ai图
ChatGPT4网页空白:重新定义智能交互的未来,怎么登ai
打破创作边界,无限可能无限制生成文章的AI
为什么选择B站VIP?让你畅享更丰富的视听体验,小门类网站seo