JS进阶系列之this (call、apply、bind)
在javascript中,this的指向是在执行上下文的创建阶段确定的,其实只要知道不同执行方式下,this的指向分别是是什么,就能很好的掌握this这个让人摸不透的东西。
一、全局执行
全局执行又分为浏览器和node下的执行环境
1、浏览器
console.log(this);//window
2、node
console.log(this);//global
在浏览器器下全局执行的this指向window,而在node环境下全局执行的this指向global
二、函数执行
函数执行又分为纯粹的函数调用还有严格模式下的函数调用
1、纯粹的函数调用
function test(){
console.log(this);
}
test();//window
2、严格模式下的函数调用
'use strict'
function test(){
console.log(this);
}
test();//undefined
纯粹的函数调用的时候,this默认在全局执行,所以指向window,但是在严格模式下,this指向undefined
三、作为对象的方法调用
当作为对象的方法调用的时候,this指向当前的对象
var obj = {
name:'McRay';
foo:function(){
console.log(this.name);
}
}
obj.foo();//McRay
我们还可以这样写
function test(){
console.log(this.name);
}
var obj = {
name:'McRay';
foo:test;
}
obj.foo();//McRay
对于函数test来说,函数名test是一个引用,foo:test这句话的作用就是让foo也指向test所指向的函数,所以它们调用的是同一个函数,所以this指向的是foo所在的对象。
再来看一种情况,把对象的方法赋值给另一个变量,然后直接调用这个变量
var obj = {
name:'McRay';
foo:function(){
console.log(this);
}
}
var test = obj.foo();
test();//window
}
我们发现此时的this指向的是全局环境下的window,因为obj.foo()只是一个函数引用,让test等于这个引用,就与obj这个对象没有了关系了,所以当调用test()的时候,其实就是全局执行,this就指向window。
上面的问题在平时的编程中也会遇到,其中比较经典的就是异步回调函数的调用,请看下面的例子
var obj = {
name:'McRay';
foo:function(){
console.log(this);
},
foo2:function(){
console.log(this);//obj
setTimeout(this.foo,1000);//window
}
};
obj.foo2();
foo2()方法中的this第一次指向obj,但是为什么第二次this就指向window呢?道理其实和上一个问题一样,setTimeout函数的第一个参数的接受一个函数,好比如fun=this.foo(),所以相当于把this.foo()这个函数引用赋给了fun,已经摆脱了obj对象的关系了,所以相当于在全局环境下执行。
解决方法是利用闭包的特性,代码如下
var obj = {
name:'McRay';
foo:function(){
console.log(this);
}
foo2:function(){
console.log(this);
var that = this;
setTimeout(function(){
console.log(this);//window
console.log(that);//obj
},1000);
}
}
obj.foo2();
四、作为一个构造函数调用
在之前的文章里面,我已经总结过,在调用构造函数(new),创建一个对象的过程都发生了什么,这里简单回顾一下,主要以下几个步骤:
- 1、创建一个空对象
- 2、让这个空对象的原型指向构造函数的原型
- 3、让构造函数中的this指向这个空对象
- 4、返回这个新的对象
所以我们在实例化一个对象的过程中,其实就是将this指向该实例化出来的新对象。
function Person(name){
this.name = name;
console.log(this);
}
var p = new Person('McRay');//p
五、箭头函数
ES6中新增加的箭头函数,让函数体内的this对象,指向的就是定义时的对象,拿之前的例子说明一下。
var obj = {
name:'McRay';
foo:function(){
console.log(this);
}
foo2:function(){
console.log(this);obj
setTimeout(()=>{
console.log(this);//obj
}
}
}
obj.foo2();
当我们在setTimeout中使用箭头函数的时候,函数体中的this默认指向的就是定义这个这个函数时,所在的对象,就是obj
六、call、apply、bind
这三个方法的目的都是差不多,就是可以动态改变this的指向
1、call的第一个参数接受this指向的对象,然后是逐个传参
2、apply的第一个参数和call相同,不同的是后面传入的是参数数组
3、bind是将生成一个新的函数,将该函数的this指向我们规定的对象或者函数,并不会立即执行
JS进阶系列之this (call、apply、bind)的更多相关文章
- 【 D3.js 进阶系列 】 进阶总结
进阶系列的文章从去年10月开始写的,晃眼又是4个多月了,想在年前总结一下. 首先恭祝大家新年快乐.今年是羊年吧.前段时间和朋友聊天,聊到十二生肖里为什么没猫,我张口就道:不是因为十二生肖开会的时候猫迟 ...
- 【 D3.js 进阶系列 — 4.0 】 绘制箭头
转自:http://www.ourd3js.com/wordpress/?p=660 [ D3.js 进阶系列 — 4.0 ] 绘制箭头 发表于2014/12/08 在 SVG 绘制区域中作图,在绘制 ...
- 前端面试 js 你有多了解call,apply,bind?
函数原型链中的 apply,call 和 bind 方法是 JavaScript 中相当重要的概念,与 this 关键字密切相关,相当一部分人对它们的理解还是比较浅显,所谓js基础扎实,绕不开这些基础 ...
- 【 D3.js 进阶系列 — 1.1 】 其它表格文件的读取
CSV 表格文件是以逗号作为单元分隔符的,其他还有以制表符 Tab 作为单元分隔符的 TSV 文件,还有人为定义的其他分隔符的表格文件.本文将说明在 D3 中怎样读取它们. 1. TSV 表格文件是什 ...
- 【 D3.js 进阶系列 — 1.2 】 读取 CSV 文件时乱码的解决方法
在 D3 中使用 d3.csv 读取 CSV 文件时,有时会出现乱码问题. 怎么解决呢? 1. 乱码问题 使用 d3.csv 读取 xxx.csv 文件时.假设 xxx.csv 文件使用的是 UTF- ...
- JS进阶系列之作用域链
在之前写的进阶系列里面,提到了执行上下文在创建阶段,要创建变量对象.确定作用域链还有确定this的指向,本次将重点讲解一下作用域链. JavaScript代码的执行过程 在讲解作用域链之前,首先了解一 ...
- 使用另一种方式实现js中Function的调用(call/apply/bind)
在JavaScript中函数的调用可以有多种方式,但更经典的莫过于call和apply.call跟apply都绑定在函数上,他们两个的第一个参数意义相同,传入一个对象,他作为函数的执行环境(实质上是为 ...
- 【 D3.js 进阶系列 — 6.1 】 缩放的应用(Zoom)
缩放(Zoom)是另一种重要的可视化操作,主要是使用鼠标的滚轮进行. 1. zoom 的定义 缩放是由 d3.behavior.zoom() 定义的. var zoom = d3.behavior.z ...
- 【 D3.js 进阶系列 — 2.1 】 力学图的事件 + 顶点的固定
本章讨论在力学图中经常使用到的事件( Event ),然后对[p=555">进阶 - 第 2.0 章]的人物关系图进行改进.使用户可以固定拖拽的对象. 在[入门 - 第 9.2 章]和 ...
随机推荐
- Ant打包可运行的Jar包(加入第三方jar包)
本章介绍使用ant打包可运行的Jar包. 打包jar包最大的问题在于如何加入第三方jar包使得jar文件可以直接运行.以下用一个实例程序进行说明. 程序结构: 关键代码: package com.al ...
- 通过Hive将数据写入到ElasticSearch
我在<使用Hive读取ElasticSearch中的数据>文章中介绍了如何使用Hive读取ElasticSearch中的数据,本文将接着上文继续介绍如何使用Hive将数据写入到Elasti ...
- JDBC (29)
1.JDBC:就是一套API,由sun公司定义类或者定义的接口.(全称 java database connectivity:是Java访问数据库的标准规范),Java提供访问数据库规范称为JDBC, ...
- javascript实现异步编程的4种方法
1.回调函数. 2.事件监听 . 思路:采用事件驱动模式.任务的执行不取决于代码的顺序,而取决于某个事件是否发生 3.观察者模式 (发布/订阅模式) 代码如下: jQuery.subscribe ...
- JVM探秘4---垃圾收集器介绍
Java虚拟机有很多垃圾收集器 下面先来了解HotSpot虚拟机中的7种垃圾收集器:Serial.ParNew.Parallel Scavenge.Serial Old.Parallel Old.CM ...
- C# 数值类型和无穷大
在c#语言中的数字有两个特性要了解.例如:任何数除以0所得的结果是无穷大,不在int long 和decimal类型的范围内.所以计算(一个数除以0会出错),但是在double和float类型中有一个 ...
- System.getSecurityManager()
https://www.cnblogs.com/yiwangzhibujian/p/6207212.html java安全管理器SecurityManager入门 一.文章的目的 这是一篇对Jav ...
- httpclient get post
https://www.cnblogs.com/wutongin/p/7778996.html post请求方法和get请求方法 package com.xkeshi.paymentweb.contr ...
- calendar 模块
calendar模块,即日历模块,提供了对日期的一些操作方法,和生成日历的方法 注:星期一是默认的每周第一天,星期天是默认的最后一天.更改设置需调用calendar.setfirstweekday() ...
- scala 操作hdfs
获取hdfs文件下所有文件getAllFiles 遍历 spark读取 1 package com.spark.demo import java.io.IOException import java. ...