在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. uboot中fdt命令的使用

    转载:https://blog.csdn.net/voice_shen/article/details/7441894 依linux community的要求,从linux-3.5后,新提交的code ...

  2. 【upstream】Nginx配置upstream实现负载均衡

    如果Nginx没有仅仅只能代理一台服务器的话,那它也不可能像今天这么火,Nginx可以配置代理多台服务器,当一台服务器宕机之后,仍能保持系统可用.具体配置过程如下: 1. 在http节点下,添加ups ...

  3. 改变 select下拉框 样式

    select{ outline: none; text-indent: 10px; height: 45px; line-height: 45px; width: 100%; border:1px s ...

  4. linux常用命令:iostat 命令

    Linux系统中的 iostat 是I/O statistics(输入/输出统计)的缩写,iostat工具将对系统的磁盘操作活动进行监视.它的特点是汇报磁盘活动统计情况,同时也会 汇报出CPU使用情况 ...

  5. JUC原子类 1

    根据修改的数据类型,可以将JUC包中的原子操作类可以分为4类. 1. 基本类型: AtomicInteger, AtomicLong, AtomicBoolean ; 2. 数组类型: AtomicI ...

  6. ab命令压力测试

    网站性能压力测试是服务器网站性能调优过程中必不可缺少的一环.只有让服务器处在高压情况下,才能真正体现出软件.硬件等各种设置不当所暴露出的问题. 性能测试工具目前最常见的有以下几种:ab.http_lo ...

  7. python 操作 hbase

    python 是万能的,当然也可以通过api去操作big database 的hbase了,python是通过thrift去访问操作hbase 以下是在centos7 上安装操作,前提是hbase已经 ...

  8. Low-level Native Plugin Interface

    http://docs.manew.com/Manual/index.htm https://docs.unity3d.com/Manual/NativePluginInterface.html ht ...

  9. Mysql管理工具 SqlYog快捷键大全

    Ctrl+M   创建一个新的连接Ctrl+N   使用当前设置新建连接Ctrl+F4   断开当前连接 对象浏览器F5   刷新对象浏览器(默认)Ctrl+B   设置焦点于对象浏览器 SQL 窗口 ...

  10. linux不常用但很有用的命令(持续完善)

    Linux登录后设置提示信息: /etc/issue 本地端登录前显示信息文件 /etc/issue.net 网络端登录前显示信息文件 /etc/motd 登陆后显示信息文件 可以添加以下几个常用选项 ...