关于this的全面解析(call,apply,new)
我们在写代码的时候,时常会被this弄的傻傻分不清楚,看源码的时候也经常被call啊apply啊弄的头皮发麻。this到底是什么?本文主要根据书上和实际应用做了一些归纳。一般情况下this有4种绑定规则:
1、默认绑定 - this指向全局变量
举例:
function baz(){
console.log('a',this.a);
};
var a = 2;
baz();
// a 2
在非严格模式下,我们调用了baz,它没有被任何对象包裹,而是暴露在全局环境中,因此此时它被调用的上下文就是全局环境,所以获取到this.a是2;
2、隐式绑定
这种情况应该是最常见的,我们调用的方法被包含在某个对象中,此时应该找到调用这个方法的对象所处的上下文。
举例:
function baz(){
console.log('a',this.a);
};
var a = 2;
var obj = {
a: 3,
baz: baz
}
obj.baz();
// a, 3
这个时候,this指向调用baz这个方法的对象obj;所以获取的this.a就是obj.a是3;
思考一下下面的this.a的值又是什么:
function baz(){
console.log('a',this.a);
};
var a = 2;
var obj = {
a: 3,
baz: baz
};
var obj2 = {
a: 4,
baz: obj.baz
}
obj2.baz();
tips: 对象属性的引用链只有最后一层在调用位置中起作用。
3、显式绑定
这里就是函数可以使用的方法call(obj,param1,param2,...)
和apply(obj,[param1,param2,...])
,它们的作用是我们可以显式的改变函数的上下文(this),参数说明:
- obj:一个对象,将方法的this指向该对象的this;
- paramX:参数,两者的表现形式不同
var a = 2;
function baz(){
console.log('a',this.a);
};
var obj = {
a: 3
};
baz.call(obj); // 改变了this,并执行了函数
// a 3
如果直接baz(),则采用的是第一种规则,这里使用baz.call强行将baz的this指向了obj,所以此时baz中的this是obj;
应用场景:带参数的
var a = 2;
function baz(p1, p2) {
console.log(this.a, p1, p2);
};
var obj = {
a: 3
};
var foo = function () {
return baz.apply(obj, arguments)
};
foo("hi,", "heimayu");
// 3 hi, heimayu
4、new绑定
在js中,我们经常使用new来对函数进行构造调用,如下:
function Foo(a) {
this.a = a;
};
var foo = new Foo(3);
console.log("a", foo.a)
这里,使用new Foo()的时候构造出一个新的对象foo并把它绑定到Foo调用中的this上。
5、优先级
在大多数情况下我们找到函数的调用位置,并判断应用哪种规则,就可以快速找到this;结论:
new绑定 > 显式绑定(call,apply) > 隐式绑定 > 全局默认绑定
6、一些例外
规则是死的,程序是活的,总有例外出现。比如啊:
我不关心this是什么
function foo(a, b) {
console.log("a:" + a, "b:" + b);
};
foo.apply(null, [1, 2]);
// a:1 b:2
ES6箭头函数
箭头函数不使用上面的规则,而是根据 由外层的作用域来决定它的this
function foo() {
// return function () {
// console.log('第一种情况', this.a)
// }
return (a)=>{
console.log('第二种情况',this.a)
}
};
var obj1 = {
a: 2
};
var obj2 = {
a: 3
};
var bar = foo.call(obj1);
bar.call(obj2);
// 第一种情况:3
// 第二种情况:2
记住箭头函数的绑定无法被修改!!!因此在执行bar的时调用了foo,而foo中的this是指向到obj1的。
最常见的场景:
var a = 1;
var obj = {
a:2,
foo: function(){
setTimeout(function(){
console.log(this.a)
},100)
}
};
obj.foo();
// 1
在ES5中,我们想打出的值是2,会这样做:
var a = 1;
var obj = {
a:2,
foo: function(){
let self = this;
setTimeout(function(){
console.log(self.a)
},100)
}
};
obj.foo();
在ES6中,则可以直接这样:
var a = 1;
var obj = {
a:2,
foo: function(){
setTimeout(()=>{
console.log(this.a)
},100)
}
};
obj.foo();
理解一下:因为箭头函数中的this,指向了foo,foo的上下文是obj。具体的说:箭头函数会继承外层函数调用的this绑定。
关于this的全面解析(call,apply,new)的更多相关文章
- 完全解析Array.apply(null, { length: 1000 })
Array.apply(null, { length: 1000 }) 点击打开视频讲解更加详细 在阅读VueJS教程时有这么段demo code: render: function (createE ...
- Spring mybatis源码篇章-NodeHandler实现类具体解析保存Dynamic sql节点信息
前言:通过阅读源码对实现机制进行了解有利于陶冶情操,承接前文Spring mybatis源码篇章-XMLLanguageDriver解析sql包装为SqlSource SqlNode接口类 publi ...
- Mybatis源码解析-DynamicSqlSource和RawSqlSource的区别
XMLLanguageDriver是ibatis的默认解析sql节点帮助类,其中的方法其会调用生成DynamicSqlSource和RawSqlSource这两个帮助类,本文将对此作下简单的简析 应用 ...
- Android进阶:七、Retrofit2.0原理解析之最简流程【下】
紧接上文Android进阶:七.Retrofit2.0原理解析之最简流程[上] 一.请求参数整理 我们定义的接口已经被实现,但是我们还是不知道我们注解的请求方式,参数类型等是如何发起网络请求的呢? 这 ...
- Spring源码分析(七)bean标签的解析及注册
摘要:本文结合<Spring源码深度解析>来分析Spring 5.0.6版本的源代码.若有描述错误之处,欢迎指正. 在上一篇中提到过Spring中的标签包括默认标签和自定义标签两种,而两种 ...
- InnoDB recovery过程解析
本文来自网易云社区. InnoDB如果发生意外宕机了,数据会丢么?对于这个问题,稍微了解一点MySQL知识的人,都会斩钉截铁的回答:不会!为什么?他们也会毫不犹豫的说:因为有重做日志(redo log ...
- Mybaits 源码解析 (六)----- 全网最详细:Select 语句的执行过程分析(上篇)(Mapper方法是如何调用到XML中的SQL的?)
上一篇我们分析了Mapper接口代理类的生成,本篇接着分析是如何调用到XML中的SQL 我们回顾一下MapperMethod 的execute方法 public Object execute(SqlS ...
- Gradle--初识
1.Eclipse从svn导入Gradle项目 1.检出项目的时候不要选新项目,选"做为工作空间中的项目检出",然后点Finish. 2.将项目转为Gradle项目,右键导入的项目 ...
- MyBatis 源码分析 - SQL 的执行过程
* 本文速览 本篇文章较为详细的介绍了 MyBatis 执行 SQL 的过程.该过程本身比较复杂,牵涉到的技术点比较多.包括但不限于 Mapper 接口代理类的生成.接口方法的解析.SQL 语句的解析 ...
- MySQL · 引擎特性 · 基于InnoDB的物理复制实现(转载)
http://mysql.taobao.org/monthly/2016/05/01/ 在开始之前,你需要对InnoDB的事务系统有个基本的认识.如果您不了解,可以参考我之前的几篇关于InnoDB的文 ...
随机推荐
- 队列的理解和实现(二) ----- 链队列(java实现)
什么是链队列 链队是指采用链式存储结构实现的队列,通常链队用单链表俩表示.一个链队显然需要两个分别指示队头和队尾的指针,也称为头指针和尾指针,有了这两个指针才能唯一的确定. package 链队列; ...
- 使用Chrome-headless模式下,截屏不全屏的问题
在headless模式下,是没有打开浏览器窗口的,那么driver.maximize_window(),找不到目标也打不开. 我们可以换一种方式,去在无头模式下,指定浏览器的窗口大小运行即可. __o ...
- Linux Shell编程、变量、控制语句
为什么要学习Shell编程 1)Linux运维工程师在进行服务器集群管理时,需要编写Shell程序来进行服务器管理. 2)对于JavaEE和Python程序员来说,工作的需要,你的老大会要求你编写一些 ...
- 前端知识总结--js原型链
js的原型链听着比较深奥,看着容易晕,梳理一下还是比较容易懂的 (先简单写下,后续有时间再整理) 简而言之 原型链:就是js的对象与对象之间,通过原型组成建立的层层关系,构成了整个链条,称之为原型链 ...
- springmvc 运行原理 Spring ioc的实现原理 Mybatis工作流程 spring AOP实现原理
SpringMVC的工作原理图: SpringMVC流程 . 用户发送请求至前端控制器DispatcherServlet. . DispatcherServlet收到请求调用HandlerMappin ...
- MarkDown添加图片的三种方式
插图最基础的格式就是: ![Alt text](图片链接 "optional title") Alt text:图片的Alt标签,用来描述图片的关键词,可以不写.最初的本意是当图片 ...
- mono上部署web程序初体验
早就想体验一下mono,但一直琐事缠身.难得有时间,便在网上一通狂搜mono相关的资料. 如果想使用Apache服务器,只能使用mod_mono的方式,这里有详细的介绍.这种方式有点繁琐,需要安装一大 ...
- 模块和处理程序之通过HttpModule和HttpHandler拦截入站HTTP请求执行指定托管代码模块
1.简介 大多数情况下,作为一个asp.net web开发对整个web应用程序的控制是十分有限的,我们的控制往往只能做到对应用程序(高层面)的基本控制.但是,很多时候,我们需要能够低级层面进行交互,例 ...
- Asp.Net Cache缓存技术学习
本文参考自Fish Li的细说 ASP.NET Cache 及其高级用法 一.前言,相信大多数做网站开发的都知道缓存技术对于网站的重要性,它对于网站的性能优化起着至关重要的作用. 关于缓存的技术大致有 ...
- mysql 非安装版的配置
一直以来都是使用wamp中集成的mysql数据库,今天突然想试试下载一个mysql的zip包进行配置. 一.下载mysql非安装版 下载地址可以到:http://dev.mysql.com/downl ...