JavaScript基础之this详解


Posted in Javascript onJune 04, 2017

JavaScript的this和Java等面向对象语言中的this大不一样,bind()、call()和apply()函数更是将this的灵活度进一步延伸。

为了保证可读性,本文采用意译而非直译。另外,本文版权归原作者所有,翻译仅用于学习。

如果对JavaScript的关键字this理解不够深刻,有时候会掉入意想不到的坑。在这里我们总结了5条通用规则来帮助你判断this到底指向什么。虽然没有囊括所有的情况,但日常大部分情况都可以使用这些规则来正确推断。

this的值通常是由所在函数的执行环境决定,也就是说要看函数是如何被调用的;
同一个函数每一次调用,this都可能指向不同的对象;

全局对象 (Global Object)

打开Chrome浏览器开发者面板(Windows: Ctrl + Shift + J)(Mac: Cmd + Option + J),并且输入:

console.log(this);

看看输出了什么?

// Window {}

window对象! 因为在全局作用域下,this指向全局对象。在浏览器中全局对象就是window对象。
为了让你更加清楚理解为什么this会指向window对象,我们来看另外一个例子:

var myName = 'Brandon';
我们可以通过在控制台输入myName来访问其值:

myName
// 输出 'Brandon'

其实,所有在全局定义的变量都绑定到window对象。我们来做如下测试:

window.myName
// 输出 'Brandon'
window.myName === myName
// 输出 true

现在我们将this放到函数内部,看看有何效果。

function test(){
 return this;
}
test();

你会发现this依然指向全局的window对象。因为this关键字没有处于一个声明的对象内部,默认指向全局window对象。这一点可能对于大部分初学者来说有点难以理解。当读完这篇文章,你就会豁然开朗。
注意:如果在strcit模式下,上面的例子中this为undefined。

声明的对象 (Declared Object)

当this关键字在一个声明对象内部使用,其值会被绑定到调用该this的函数的最近的父对象。我们用例子来说明这个问题:

var person = {
 first: 'John',
 last: 'Smith', 
 full: function() {
  console.log(this.first + ' ' + this.last);
 }
};
person.full();
// 输出 'John Smith'

在被声明的对象person中的full函数里面使用了this, 那么调用this的full函数的最近的父对象就是person, 因此,this指向person。

为了更好的描述this实际上是指向person对象的,你可以拷贝如下代码到浏览器控制台,将this打印出来。

var person = {
 first: 'John',
 last: 'Smith', 
 full: function() {
  console.log(this);
 }
};
person.full();
// 输出 Object {first: "John", last: "Smith", full: function}

我们接下来看一个更复杂的例子:

var person = {
 first: 'John',
 last: 'Smith',
 full: function() {
  console.log(this.first + ' ' + this.last);
 },
 personTwo: {
  first: 'Allison',
  last: 'Jones',
  full: function() {
   console.log(this.first + ' ' + this.last);
  }
 }
};

在这里我们有嵌套的对象,在这个时候,this分别指向谁呢? 我们将其打印出来看一看:

person.full();
// 输出 'John Smith'
person.personTwo.full();
// 输出 'Allison Jones'

你会发现满足我们前面所描述的规则:其值会被绑定到调用this的函数的最近的父对象。

new关键字

当使用new关键字构建一个新的对象,this会绑定到这个新对象。我们来看一个例子:

function Car(make, model) {
 this.make = make;
 this.model = model;
};

根据第一条规则,你可能会推断this指向全局对象。但是如果我们使用new关键字来声明一个新的变量,Car函数中的this将会绑定一个新的空对象,然后初始化this.make和this.model的值。

var myCar = new Car('Ford', 'Escape');
console.log(myCar);
// 输出 Car {make: "Ford", model: "Escape"}

call, bind, 和apply

我们可以显示的在call(),bind(),apply()中设置this的绑定对象。这三个函数很类似,但是我们需要注意它们微小的区别。

<!-- call可以接收任意个数的参数,其中第一个必须是一个this对象,其余依次是所有的参数。 -->

我们来看一个例子:

function add(c, d) {
 console.log(this.a + this.b + c + d);
}
add(3,4);
// 输出 NaN

add函数输出NaN, 因为this.a和this.b未定义。

现在我们引入对象,并且使用call()和apply()来调用:

function add(c, d) {
 console.log(this.a + this.b + c + d);
}
var ten = {a: 1, b: 2};
add.call(ten, 3, 4);
// 输出 10
add.apply(ten, [3,4]);
// 输出 10

当我们使用add.call(),第一个参数是this需要绑定的对象,剩下的是add函数本来的参数。
因此,this.a指向ten.a,this.b指向ten.b。add.apply()也类似,除了第二个参数是一个数组,用于存储add函数的参数。

bind()函数和call()类似,但是bind()函数不会立即被调用。bind()函数会返回一个函数,并且将this绑定好。接下来我们来用例子来帮助理解bind()函数的应用场景:

var small = {
 a: 1,
 go: function(b,c,d){
  console.log(this.a+b+c+d);
 }
}
var large = {
 a: 100
}

执行:

small.go(2, 3, 4);
// 输出 10

如果我们想使用large.a的值,而不是small.a呢? 我们可以使用call/apply:

small.go.call(large, 2, 3, 4);
// 输出 109

但是,如果我们现在还不知道这三个参数应该传入什么值,应该怎么办呢? 我们可以使用bind:

var bindTest = small.go.bind(large, 2);

如果我们将bindTest在控制台下打印出来,我们会看到:

console.log(bindTest);
// 输出 function (b,c,d){console.log(this.a+b+c+d);}

注意:该函数已经将this绑定到large对象,并且传入了第一个参数b。所以,我们接下来是需要传入余下的参数即可:

bindTest(3, 4);
// 输出 109

箭头函数(=>)

因为需要很大的篇幅,我们会专门写一篇博客来介绍。

结论

当你读完这篇博客,你应该可以理解大多数情况下this指向的对象。
接下来我们来总结一下:

this的值通常是由当前函数的执行环境所决定;
在全局作用域,this指向全局对象 (window对象);
当使用new关键字声明,this指向新建对象;
我们可以使用call(), bind(), apply()来设置this;
箭头函数不会绑定this。

原文: JavaScript: The Keyword ‘This' for Beginners

译者: Fundebug

Javascript 相关文章推荐
常用参考资料(手册)下载或者链接
Jul 22 Javascript
用htc组件制作windows选项卡
Jan 13 Javascript
javascript判断单选框或复选框是否选中方法集锦
Apr 04 Javascript
JavaScript 闭包在封装函数时的简单分析
Nov 28 Javascript
jquery datepicker参数介绍和示例
Apr 15 Javascript
JS实现从连接中获取youtube的key实例
Jul 02 Javascript
详解页面滚动值scrollTop在FireFox与Chrome浏览器间的兼容问题
Dec 03 Javascript
简单实现js放大镜效果
Jul 24 Javascript
JS实现留言板功能[楼层效果展示]
Dec 27 Javascript
js实现移动端吸顶效果
Jan 08 Javascript
前端如何实现动画过渡效果
Feb 05 Javascript
详解vue身份认证管理和租户管理
May 25 Vue.js
Angular 4 依赖注入学习教程之FactoryProvider的使用(四)
Jun 04 #Javascript
Angular 4依赖注入学习教程之ClassProvider的使用(三)
Jun 04 #Javascript
Angular 4依赖注入学习教程之组件服务注入(二)
Jun 04 #Javascript
JavaScript箭头(arrow)函数详解
Jun 04 #Javascript
Angular 4依赖注入学习教程之简介(一)
Jun 04 #Javascript
angularJs中datatable实现代码
Jun 03 #Javascript
angularJS利用ng-repeat遍历二维数组的实例代码
Jun 03 #Javascript
You might like
php中用于检测一个地理IP地址是否可用的代码
2012/02/19 PHP
php实现简单加入购物车功能
2017/03/07 PHP
PHP实现的登录页面信息提示功能示例
2017/07/24 PHP
PHP var关键字相关原理及使用实例解析
2020/07/11 PHP
彪哥1.1(智能表格)提供下载
2006/09/07 Javascript
jQuery 拖动层(在可视区域范围内)
2012/05/24 Javascript
FF火狐下获取一个元素同类型的相邻元素实现代码
2012/12/15 Javascript
利用js制作html table分页示例(js实现分页)
2014/04/25 Javascript
JavaScript Math.ceil 方法(对数值向上取整)
2015/01/09 Javascript
js下拉选择框与输入框联动实现添加选中值到输入框的方法
2015/08/17 Javascript
javascript 继承学习心得总结
2016/03/17 Javascript
vue实现简单表格组件实例详解
2017/04/16 Javascript
详解使用PM2管理nodejs进程
2017/10/24 NodeJs
通过vue-cli3构建一个SSR应用程序的方法
2018/09/13 Javascript
vue实现微信二次分享以及自定义分享的示例
2019/03/20 Javascript
浅谈一种让小程序支持JSX语法的新思路
2019/06/16 Javascript
Vue项目接入Paypal实现示例详解
2020/06/04 Javascript
jQuery实现开关灯效果
2020/08/02 jQuery
[54:29]2018DOTA2亚洲邀请赛 4.7 淘汰赛 VP vs LGD 第二场
2018/04/09 DOTA
[03:08]迎霜节狂欢!2018年迎霜节珍藏Ⅰ一览
2018/12/25 DOTA
Python中Collections模块的Counter容器类使用教程
2016/05/31 Python
python中执行shell的两种方法总结
2017/01/10 Python
Python实现获取照片拍摄日期并重命名的方法
2017/09/30 Python
Python读取系统文件夹内所有文件并统计数量的方法
2018/10/23 Python
python 为什么说eval要慎用
2019/03/26 Python
Python散点图与折线图绘制过程解析
2019/11/30 Python
Python如何基于smtplib发不同格式的邮件
2019/12/30 Python
哈工大自然语言处理工具箱之ltp在windows10下的安装使用教程
2020/05/07 Python
python 实现简易的记事本
2020/11/30 Python
PHP如何防止SQL注入
2014/05/03 面试题
构造方法和其他方法的区别?怎么调用父类的构造方法
2013/09/22 面试题
毕业生毕业总结的自我评价范文
2013/11/02 职场文书
物业管理工作方案
2014/05/10 职场文书
习总书记三严三实学习心得体会
2014/10/13 职场文书
python使用pygame创建精灵Sprite
2021/04/06 Python
python中如何对多变量连续赋值
2021/06/03 Python