this的四种绑定规则总结
一、默认绑定
1、全局环境中,this默认绑定到window
2、函数独立调用时,this默认绑定到window
console.log(this === window);//true
function foo(){
console.log(this === window);
}
foo(); //true
3、被嵌套的函数独立调用时,this默认绑定到window
var a = ;
var obj = {
a : ,
foo:function(){
function test(){
console.log(this.a);
}
test(); //虽然test()函数被嵌套在obj.foo()函数中,但test()函数是独立调用,而不是方法调用。所以this默认绑定到window
}
}
obj.foo();//
4、IIFE立即执行函数实际上是函数声明后直接调用执行
var a = ;
function foo(){
(function test(){
console.log(this.a);
})()
};
var obj = {
a : ,
foo:foo
}
obj.foo();//0 //等价于上例3
var a = ;
var obj = {
a : ,
foo:function(){
function test(){
console.log(this.a);
}
test();
}
}
obj.foo();//
5、【闭包】类似地,test()函数是独立调用,而不是方法调用,所以this默认绑定到window
var a = ;
function foo(){
function test(){
console.log(this.a);
}
return test;
};
var obj = {
a : ,
foo:foo
}
obj.foo()();//0,实际上就是test()方法独立调用,所以this是window
闭包里面this重定向:由于闭包的this默认绑定到window对象,但又常常需要访问嵌套函数的this,所以常常在嵌套函数中使用var that = this,然后在闭包中使用that替代this,使用作用域查找的方法来找到嵌套函数的this值
var a = ;
function foo(){
var that = this;
function test(){
console.log(that.a);
}
return test;
};
var obj = {
a : ,
foo:foo
}
obj.foo()();//
obj.foo()是obj对象的foo方法的调用,所以里面的this指向obj,用that来重定向this使that指向obj,然后obj.foo()()实际上就是test()方法独立调用,但是是that.a,此时that已经指向了obj,所以是2
二、隐式绑定
一般地,被直接对象所包含的函数调用时,也称为方法调用,this隐式绑定到该直接对象
function foo(){
console.log(this.a);
};
var obj1 = {
a:,
foo:foo,
obj2:{
a:,
foo:foo
}
}
//foo()函数的直接对象是obj1,this隐式绑定到obj1
obj1.foo();//1
//foo()函数的直接对象是obj2,this隐式绑定到obj2
obj1.obj2.foo();//
隐式丢失
隐式丢失是指被隐式绑定的函数丢失绑定对象,从而默认绑定到window。这种情况容易出错却又常见
1、【函数别名】
var a = ;
function foo(){
console.log(this.a);
};
var obj = {
a : ,
foo:foo
}
//把obj.foo赋予别名bar,造成了隐式丢失,因为只是把foo()函数赋给了bar,而bar与obj对象则毫无关系
var bar = obj.foo;
bar();//0 //等价于
var a = ;
var bar = function foo(){
console.log(this.a);
}
bar();//
2、【参数传递】
var a = ;
function foo(){
console.log(this.a);
};
function bar(fn){
fn();
}
var obj = {
a : ,
foo:foo
}
//把obj.foo当作参数传递给bar函数时,有隐式的函数赋值fn=obj.foo。与上例类似,只是把foo函数赋给了fn,而fn与obj对象则毫无关系
bar(obj.foo);//
//等价于
var a = ;
function bar(fn){
fn();
}
bar(function foo(){
console.log(this.a);
});
3、【内置函数】内置函数与上例类似,也会造成隐式丢失
var a = ;
function foo(){
console.log(this.a);
};
var obj = {
a : ,
foo:foo
}
setTimeout(obj.foo,);//0 //等价于
var a = ;
setTimeout(function foo(){
console.log(this.a);
},);//
因此,内置函数里也常常需要 this 重定向
4、【间接引用】函数的"间接引用"一般都在无意间创建,最容易在赋值时发生,会造成隐式丢失
function foo() {
console.log( this.a );
}
var a = ;
var o = { a: , foo: foo };
var p = { a: };
o.foo(); // 3
//将o.foo函数赋值给p.foo函数,然后立即执行。相当于仅仅是foo()函数的立即执行
(p.foo = o.foo)(); //
要与下面这种情况进行区分:
function foo() {
console.log( this.a );
}
var a = ;
var o = { a: , foo: foo };
var p = { a: };
o.foo(); // 3
//将o.foo函数赋值给p.foo函数,之后p.foo函数再执行,是属于p对象的foo函数的执行
p.foo = o.foo;
p.foo();//
5、【其他情况】
在javascript引擎内部,obj和obj.foo储存在两个内存地址,简称为M1和M2。只有obj.foo()这样调用时,是从M1调用M2,因此this指向obj。但是,下面三种情况,都是直接取出M2进行运算,然后就在全局环境执行运算结果(还是M2),因此this指向全局环境
var a = ;
var obj = {
a : ,
foo:foo
};
function foo() {
console.log( this.a );
}; (obj.foo = obj.foo)();// (false || obj.foo)();// (, obj.foo)();//
三、显式绑定
通过call()、apply()、bind()方法把对象绑定到this上,叫做显式绑定。对于被调用的函数来说,叫做间接调用
var a = ;
function foo(){
console.log(this.a);
}
var obj = {
a:
};
foo();//
foo.call(obj);//
普通的显式绑定无法解决隐式丢失问题
1、【硬绑定】硬绑定是显式绑定的一个变种,使this不能再被修改
var a = ;
function foo(){
console.log(this.a);
}
var obj = {
a:
};
var bar= function(){
foo.call(obj);
}
//在bar函数内部手动调用foo.call(obj)。因此,无论之后如何调用函数bar,它总会手动在obj上调用foo
bar();//
setTimeout(bar,);//
bar.call(window);//
2、【API】javascript中新增了许多内置函数,具有显式绑定的功能,如数组的:map()、forEach()、filter()、some()、every()
var id = 'window';
function foo(el){
console.log(el,this.id);
}
var obj = {
id: 'fn'
};
[,,].forEach(foo);//1 "window" 2 "window" 3 "window"
[,,].forEach(foo,obj);//1 "fn" 2 "fn" 3 "fn"
四、new绑定
如果函数或者方法调用之前带有关键字new,它就构成构造函数调用。对于this绑定来说,称为new绑定
1、构造函数通常不使用return关键字,它们通常初始化新对象,当构造函数的函数体执行完毕时,它会显式返回。在这种情况下,构造函数调用表达式的计算结果就是这个新对象的值
function fn(){
this.a = ;
}
var test = new fn();
console.log(test);//{a:2}
2、如果构造函数使用return语句但没有指定返回值,或者返回一个原始值,那么这时将忽略返回值,同时使用这个新对象作为调用结果
function fn(){
this.a = ;
return ;
}
var test = new fn();
console.log(test);//{a:2}
3、如果构造函数显式地使用return语句返回一个对象,那么调用表达式的值就是这个对象
var obj = {a:};
function fn(){
this.a = ;
return obj;
}
var test = new fn();
console.log(test);//{a:1}
【注意】尽管有时候构造函数看起来像一个方法调用,它依然会使用这个新对象作为this。也就是说,在表达式new o.m()中,this并不是o,而是新对象
var o = {
m: function(){
return this;
}
}
var obj = new o.m();
console.log(obj,obj === o);//{} false
console.log(obj.constructor === o.m);//true
var o = {
m: function(){
this.a = ;
return this;
}
}
var obj = new o.m();
console.log(obj,obj === o);//o.m {a: 2} false
console.log(obj.constructor === o.m);//true
五、严格模式
1、严格模式下,独立调用的函数的this指向undefined
function fn(){
'use strict';
console.log(this);//undefined
}
fn();
function fn(){
console.log(this);//window
}
fn();
2、在非严格模式下,使用函数的call()或apply()方法时,null或undefined值会被转换为全局对象。而在严格模式下,函数的this值始终是指定的值
var color = 'red';
function displayColor(){
console.log(this.color);
}
displayColor.call(null);//red var color = 'red';
function displayColor(){
'use strict';
console.log(this.color);
}
displayColor.call(null);//TypeError: Cannot read property 'color' of null
this的四种绑定规则:默认绑定、隐式绑定、显式绑定和new绑定,分别对应函数的四种调用方式:独立调用、方法调用、间接调用和构造函数调用。分清这四种绑定规则不算难,比较麻烦的是需要练就火眼金睛,识别出隐式丢失的情况。
说到底,javascript如此复杂的原因是因为函数过于强大。因为,函数是对象,所以原型链比较复杂;因为函数可以作为值被传递,所以执行环境栈比较复杂;同样地,因为函数具有多种调用方式,所以this的绑定规则也比较复杂。只有理解了函数,才算理解了javascript。
this的四种绑定规则总结的更多相关文章
- 深入理解this机制系列第一篇——this的4种绑定规则
× 目录 [1]默认绑定 [2]隐式绑定 [3]隐式丢失[4]显式绑定[5]new绑定[6]严格模式 前面的话 如果要问javascript中哪两个知识点容易混淆,作用域查询和this机制绝对名列前茅 ...
- JS中this的4种绑定规则
this ES6中的箭头函数采用的是词法作用域. 为什么要使用this:使API设计得更简洁且易于复用. this即不指向自身,也不指向函数的词法作用域. this的指向只取决于函数的调用方式 thi ...
- this的四种绑定形式
一 , this的默认绑定 当一个函数没有明确的调用对象的时候,也就是单纯作为独立函数调用的时候,将对函数的this使用默认绑定:绑定到全局的window对象. 一个例子 function fire ...
- this四种绑定方式之间的奇淫技巧
写在前面 上一篇中,我们对于JavaScript中原始值.复杂值以及内存空间进行了一个深入浅出的总结,这次我们来聊一聊JavaScript中this关键字的深入浅出的用法. 在 JavaScript ...
- ASP.NET Eval四种绑定方式
1.1.x中的数据绑定语法 <asp:Literal id="litEval2" runat="server" Text='<%#DataBinde ...
- 【javascript】函数中的this的四种绑定形式
目录 this的默认绑定 this的隐式绑定 隐式绑定下,作为对象属性的函数,对于对象来说是独立的 在一串对象属性链中,this绑定的是最内层的对象 this的显式绑定:(call和bind方法) n ...
- 【javascript】函数中的this的四种绑定形式 — 大家准备好瓜子,我要讲故事啦~~
javascript中的this和函数息息相关,所以今天,我就给大家详细地讲述一番:javascript函数中的this 一谈到this,很多让人晕晕乎乎的抽象概念就跑出来了,这里我就只说最 ...
- Android studio button 按钮 四种绑定事件的方法
package com.geli_2.sujie.sujiegeili2testbutton; import android.os.Bundle; import android.support.v7. ...
- ASP.NET Eval四种绑定方式 及详解
1.1.x中的数据绑定语法 <asp:Literal id="litEval2" runat="server" Text='<%#DataBinde ...
随机推荐
- CF995E Number Clicker 解题报告
CF995E Number Clicker 题目描述 Allen is playing Number Clicker on his phone. He starts with an integer u ...
- [经验分享]Windows系统下生成IOS证书
我使用ApiCloud开发APP,开发后需要生成IOS的证书才能在项目开发控制台中进行编译,于是我在网上大海捞针似的寻找办法. 官方文档提供了使用苹果系统下生成IOS证书的步骤,对于我这个没有imac ...
- Java面试题之Oracle 支持哪三种事务隔离级别
Oracle 支持三种事务隔离级别: 1.读已提交:(默认) 2.串行化: 3.只读模式
- classpath: spring 中的查找方式
Spring可以通过指定classpath*:与classpath:前缀加路径的方式从classpath加载文件,如bean的定义文件.classpath*:的出现是为了从多个jar文件中加载相同的文 ...
- Matlab 几种卷积的实现与比较(conv与filter,conv2与filter2)
Matlab 几种卷积的实现与比较(conv与filter,conv2与filter2) 最近在做控制算法实现的时候,对于其中参杂的各种差分.卷积很头疼,就在网上搜集了些资料,汇总于此,以做备 ...
- 51Nod 1601 完全图的最小生成树计数
题目链接 分析: 这是一张完全图,并且边的权值是由点的权值$xor$得到的,所以我们考虑贪心的思想,考虑$kruskal$的过程选取最小的边把两个连通块合并,所以我们可以模仿$kruskal$的过程, ...
- 【IDEA】IDEA中更新Maven的索引与像Eclipse那样搜索坐标
1.添加坐标,像Eclipse那样搜索: 右击项目中pom.xml中,然后选择Generate(快捷键是Alt+Insert)->Dependency之后输入关键字查询 或者在pom.xml中直 ...
- 大(NOIP模拟赛Round #10)
题目描述: 小Z有个n个点的高清大图,每个点有且只有一条单向边的出边.现在你可以翻转其中的一些边,使他从任何一个点都不能通过一些道路走回这个点.为了方便,你只需输出方案数对取模即可.当在两个方案中有任 ...
- hdu 5146(水题)
Sequence Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Su ...
- React Native WebView关闭缓存
React Native WebView关闭缓存 网上搜索没有找到关闭React Native下webview控件的缓存的方法,经测试找到解决方案,记录如下 核心思路:通过请求时设置请求头,使页面缓存 ...