在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的更多相关文章

  1. 【 D3.js 进阶系列 】 进阶总结

    进阶系列的文章从去年10月开始写的,晃眼又是4个多月了,想在年前总结一下. 首先恭祝大家新年快乐.今年是羊年吧.前段时间和朋友聊天,聊到十二生肖里为什么没猫,我张口就道:不是因为十二生肖开会的时候猫迟 ...

  2. 【 D3.js 进阶系列 — 4.0 】 绘制箭头

    转自:http://www.ourd3js.com/wordpress/?p=660 [ D3.js 进阶系列 — 4.0 ] 绘制箭头 发表于2014/12/08 在 SVG 绘制区域中作图,在绘制 ...

  3. 【 D3.js 进阶系列 — 1.1 】 其它表格文件的读取

    CSV 表格文件是以逗号作为单元分隔符的,其他还有以制表符 Tab 作为单元分隔符的 TSV 文件,还有人为定义的其他分隔符的表格文件.本文将说明在 D3 中怎样读取它们. 1. TSV 表格文件是什 ...

  4. 【 D3.js 进阶系列 — 1.2 】 读取 CSV 文件时乱码的解决方法

    在 D3 中使用 d3.csv 读取 CSV 文件时,有时会出现乱码问题. 怎么解决呢? 1. 乱码问题 使用 d3.csv 读取 xxx.csv 文件时.假设 xxx.csv 文件使用的是 UTF- ...

  5. JS进阶系列之作用域链

    在之前写的进阶系列里面,提到了执行上下文在创建阶段,要创建变量对象.确定作用域链还有确定this的指向,本次将重点讲解一下作用域链. JavaScript代码的执行过程 在讲解作用域链之前,首先了解一 ...

  6. 【 D3.js 进阶系列 — 6.1 】 缩放的应用(Zoom)

    缩放(Zoom)是另一种重要的可视化操作,主要是使用鼠标的滚轮进行. 1. zoom 的定义 缩放是由 d3.behavior.zoom() 定义的. var zoom = d3.behavior.z ...

  7. 【 D3.js 进阶系列 — 2.1 】 力学图的事件 + 顶点的固定

    本章讨论在力学图中经常使用到的事件( Event ),然后对[p=555">进阶 - 第 2.0 章]的人物关系图进行改进.使用户可以固定拖拽的对象. 在[入门 - 第 9.2 章]和 ...

  8. JS进阶系列之内存空间

    也许很多人像我一样,觉得JS有垃圾回收机制,内存就可以不管了,以至于在全局作用域下定义了很多变量,自以为JS会自动回收,直到最近,看了阮一峰老师,关于javascript内存泄漏的文章时,才发现自己写 ...

  9. 【 D3.js 进阶系列 — 1.0 】 CSV 表格文件的读取

    在入门系列的教程中.我们经常使用 d3.json() 函数来读取 json 格式的文件.json 格式非常强大.但对于普通用户可能不太适合,普通用户更喜欢的是用 Microsoft Excel 或 O ...

  10. 【 D3.js 进阶系列 — 6.2 】 饼状图的拖拽

    本文讲解稍微复杂一些的拖拽应用,即拖拽饼图的各部分. 在[入门 - 第 9.1 章]讲解了如何制作饼状图.饼状图的各部分是用具有宽度的弧线来表示的.在与用户进行交互的时候,如果每一部分都能拖拽,是很有 ...

随机推荐

  1. 006-Redis 发布订阅

    Redis 发布订阅(pub/sub)是一种消息通信模式:发送者(pub)发送消息,订阅者(sub)接收消息. Redis 客户端可以订阅任意数量的频道. 下图展示了频道 channel1 , 以及订 ...

  2. js-jquery-插件开发(二)【最终插件是最后一个,中间是过程】

    二.通过$.fn 向jQuery添加新的方法 2.1.基本格式: $.fn.pluginName = function() { //your code goes here } 说明:在$.fn后面添加 ...

  3. ubuntu 磁盘分区

    1:查看分区情况:df -h admin@iZwz92c0zpe8t65qe996ckZ:/$ df -h Filesystem Size Used Avail Use% Mounted on ude ...

  4. Gcc ------ gcc的使用简介与命令行参数说明

    gcc的使用简介与命令行参数说明 2011年06月19日 20:29:00 阅读数:10221 2011-06-19 wcdj 参考:<GNU gcc嵌入式系统开发 作者:董文军> (一) ...

  5. C# 定时器 一个简单 并且可以直接运行的Demo

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  6. iOS 网易彩票-1框架搭建

    仿网易彩票,最终要做成的效果如下: 一.分层搭建 1.新建一个项目,Lottery.只支持7.1以上坚屏. 2.将素材全部图片全部拉到相应的文件夹里. 3.选中Lottery--右键Show in F ...

  7. Lifyray笑傲江湖之API总结TextUtil

    package liferay; /** * */ import java.text.DecimalFormat; import java.text.NumberFormat; import java ...

  8. lnmp1.4 安装php fileinfo扩展 方法

    第一步:在lnmp1.4找到php安装的版本 使用命令 tar   -jxvf   php-7.1.7.tar.bz2 解压 第二步: 在解压的php-7.1.7文件夹里找到fileinfo文件夹,然 ...

  9. cf 459E

    cf459E 这题说的是 给定一个n点m条边的带边权的有向图,从中找出一条路径(可以带环),该路径包含的边数最多,并且要求路径中的权值必须严格递增,然后对边进行排序完个后采用dp去解特殊判断一下边权值 ...

  10. Python2 简明教程

    Python 由 Guido Van Rossum 在90年代初创建. 它现在是最流行的语言之一 我喜爱python是因为它有极为清晰的语法,甚至可以说,它就是可以执行的伪代码. 注意: 这篇文章针对 ...