你不知道的js中关于this绑定机制的解析[看完还不懂算我输]
前言
最近正在看《你不知道的JavaScript》,里面关于this绑定机制的部分讲的特别好,很清晰,这部分对我们js的使用也是相当关键的,并且这也是一个面试的高频考点,所以整理一篇文章分享一下这部分的内容,相信看本文的解析,你一定会有所收获的,如果喜欢的话可以点波赞/关注,支持一下。
个人博客了解一下:obkoro1.com
为什么要用this:
function identify() { console.log("Hello,I'm " + this.name); } let me = { name: "Kyle" }; let you = { name: "Reader" }; identify.call(me); // Hello,I'm Kyle identify.call(you); // Hello,I'm Reader 复制代码
这个简单的栗子,可以在不同的对象中复用函数identify
,不用针对每个对象编写一个新函数。
this解决的问题:
this提供了一种更优雅的方法来隐式'传递'一个对象的引用,因此可以将API设计得更加简洁并且易于复用。
this的四种绑定规则:
默认绑定:
规则:在非严格模式下,默认绑定的this
指向全局对象,严格模式下this
指向undefined
function foo() { console.log(this.a); // this指向全局对象 } var a = 2; foo(); // 2 function foo2() { "use strict"; // 严格模式this绑定到undefined console.log(this.a); } foo2(); // TypeError:a undefined 复制代码
默认绑定规则如上述栗子,书中还提到了一个微妙的细节:
function foo() { console.log(this.a); // foo函数不是严格模式 默认绑定全局对象 } var a = 2; function foo2(){ "use strict"; foo(); // 严格模式下调用其他函数,不影响默认绑定 } foo2(); // 2 复制代码
所以:对于默认绑定来说,决定this绑定对象的是函数体是否处于严格模式,严格指向undefined,非严格指向全局对象。
通常不会在代码中混用严格模式和非严格模式,所以这种情况很罕见,知道一下就可以了,避免某些变态的面试题挖坑。
隐式绑定:
规则:函数在调用位置,是否有上下文对象,如果有,那么this就会隐式绑定到这个对象上。
function foo() { console.log(this.a); } var a = "Oops, global"; let obj2 = { a: 2, foo: foo }; let obj1 = { a: 22, obj2: obj2 }; obj2.foo(); // 2 this指向调用函数的对象 obj1.obj2.foo(); // 2 this指向最后一层调用函数的对象 // 隐式绑定丢失 let bar = obj2.foo; // bar只是一个函数别名 是obj2.foo的一个引用 bar(); // "Oops, global" - 指向全局 复制代码
隐式绑定丢失:
隐式绑定丢失的问题:实际上就是函数调用时,并没有上下文对象,只是对函数的引用,所以会导致隐式绑定丢失。
同样的问题,还发生在传入回调函数中,这种情况更加常见,并且隐蔽,类似:
test(obj2.foo); // 传入函数的引用,调用时也是没有上下文对象。 复制代码
显式绑定:
就像我们上面看到的,如果单纯使用隐式绑定肯定没有办法得到期望的绑定,幸好我们还可以在某个对象上强制调用函数,从而将this
绑定在这个对象上。
规则:我们可以通过apply
、call
、bind
将函数中的this
绑定到指定对象上。
function foo() { console.log(this.a); } let obj = { a: 2 }; foo.call(obj); // 2 复制代码
传入的不是对象:
如果你传入了一个原始值(字符串,布尔类型,数字类型),来当做this的绑定对象,这个原始值转换成它的对象形式。
如果你把null
或者undefined
作为this的绑定对象传入call
/apply
/bind
,这些值会在调用时被忽略,实际应用的是默认绑定规则。
new绑定:
书中提到:在js中,实际上并不存在所谓的'构造函数',只有对于函数的'构造调用'。
new的时候会做哪些事情:
- 创建一个全新的对象。
- 这个新对象会被执行 [[Prototype]] 连接。
- 这个新对象会绑定到函数调用的this。
- 如果函数没有返回其他对象,那么new表达式中的函数调用会自动返回这个新对象。
规则:使用构造调用的时候,this会自动绑定在new期间创建的对象上。
function foo(a) { this.a = a; // this绑定到bar上 } let bar = new foo(2); console.log(bar.a); // 2 复制代码
this四种绑定规则的优先级
如果在某个调用位置应用了多条规则,如何确定哪条规则生效?
obj.foo.call(obj2); // this指向obj2 显式绑定比隐式绑定优先级高。 new obj.foo(); // thsi指向new新创建的对象 new绑定比隐式绑定优先级高。 复制代码
显式绑定和new绑定无法直接比较(会报错),默认绑定是不应用其他规则之后的兜底绑定所以优先级最低,最后的结果是:
显式绑定 > 隐式绑定 > 默认绑定
new绑定 > 隐式绑定 > 默认绑定
箭头函数的this指向不会使用上述的四条规则:
function foo() { return () => { console.log(this.a); }; } let obj1 = { a: 2 }; let obj2 = { a: 22 }; let bar = foo.call(obj1); // foo this指向obj1 bar.call(obj2); // 输出2 这里执行箭头函数 并试图绑定this指向到obj2 复制代码
从上述栗子可以得出,箭头函数的this规则:
- 箭头函数中的
this
继承于它外面第一个不是箭头函数的函数的this
指向。 - 箭头函数的 this 一旦绑定了上下文,就不会被任何代码改变。
结语
认真看完的话,相信你已经get到this的用法了,最后推荐一下《你不知道的JavaScript》,这本书真的很好,写的也很有趣,没看过的小伙伴抓紧入手了。
PS:目前离职中,大佬们有坑位可以介绍一下呀,base:上海长宁。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
CC框架实践(3): 让jsBridge更优雅
前言 CC框架下如何让我们的jsBridge更加优雅? jsBridge是作为js和java之间通信的桥梁,本身它的职责只是完成通信。 本文不是介绍js与java通信过程的实现,你可以使用第三方库(如:JsBridge),也可以自己来实现,或者用addJavascriptInterface,都可以。本文的侧重点是如何让我们的jsBridge不那么臃肿,实现得更优雅,更利于维护。 但在实际封装过程中,会发现需要我们需要解决很多耦合的问题: js调用的功能在其他module中,如何调用到这些功能,如何向jsbridge注册这些功能? jsbridge依赖了太多module,怎么解耦? 当js调用的功能是打开其它页面获取该页面处理后的结果并回调给js,怎么破? onResume? startActivityForResult? 一个常见的场景是:打开
- 下一篇
Java那些事—泛型通配符
Java的类型通配符,可以出现在类、方法上面。最常用的方式就是集合类,例如List,Set等类上面。 通配符类型 有泛型参数 List 有无类型标识 List< ? > 有通用的标识 List< object > 边界通配符 List 边界通配符 List 本文主要讨论的是最后的关于边界的通配符类型。 看一个例子 public static void main(String[] args) { ListbList = new ArrayList(); bList.add(new A()); bList.add(new B()); bList.add(new C()); A a3 = bList.get(0); B b3 = bList.get(0); C C3 = bList.get(0); List bExtends = new ArrayList(); bExtends.add(new A()); bExtends.add(new B()); bExtends.add(new C()); A a1 = bExtends.get(0); B b1 = bExt...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Windows10,CentOS7,CentOS8安装MongoDB4.0.16
- Windows10,CentOS7,CentOS8安装Nodejs环境
- CentOS8编译安装MySQL8.0.19
- Linux系统CentOS6、CentOS7手动修改IP地址
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- Red5直播服务器,属于Java语言的直播服务器
- SpringBoot2整合Redis,开启缓存,提高访问速度
- CentOS7,8上快速安装Gitea,搭建Git服务器
- SpringBoot2配置默认Tomcat设置,开启更多高级功能