本文尽量避免概念性的讲解,那样太抽象

所以下面以一些简单的例子,从易到难对this的用法总结

1.方法中的this会指向当前执行该方法的对象 如:

var name = "window"

var Tom = {
name:"Tom";
show:function(){
console.log(this+":"+this.name)
} } Tom.show(); //Tom

2.方法中的this不会指向声明它的对象 如下

var Bob={
name:"Bob",
show:function(){
console.log(this+":"+this.name)
}
}; var Tom={
name:"Tom",
show:Bob.show
}; Tom.show() ; //Tom

尽管console.log是Bob对象环境中声明的 ,但方法是由Tom对象调用执行

所以this总是会指向当前执行的对象,而不是声明的对象

其本质原因是:Bob.show方法赋值给Tom.show方法的时候不会把Bob对象也给绑定过去

下面是个将对象绑定过去的例子

3.嵌套对象中,this指向直接绑定它的对象

var Bob={
name:"Bob",
Tom:{
name:"Tom",
show:function(){console.log(this+":"+this.name)}
}
} Bob.Tom.show(); //Tom

4.用变量保存方法的执行结果,其执行对象仍然是Tom

var name="window";
var Tom={
name:"Tom",
show:function(){
console.log(this+":"+this.name)
}
}; var fun=Tom.show();
fun(); //Tom

此处fun所赋值的是Tom.show函数执行的结果,也就是Tom

上面的例子其实是下面所做对照的

5.用变量保存方法的内容,其执行对象是window

var name="window";
var Tom={
name:"Tom",
show:function(){
console.log(this+":"+this.name)
}
}; fun=Tom.show;
fun(); //Window

因为fun赋值的只是方法的内容,并没有绑定在Tom对象上,所以fun执行对象是window对象

6.若指明了调用方法的对象的话,this会指向调用其的对象 如下:

var name = "window";
var Bob= {
name:"Bob",
show:function(){console.log(this.name);}
}; var Tom= {name: "Tom"};
Bob.show(); //Bob
Bob.show.apply(); //window
Bob.show.apply(Tom); //Tom

当然call()也差不多类似,apply和call用于指明由哪个对象去执行该方法

7.函数作为构造器时,this会指向新建的对象,如下

function Tom(name,age){
this.name=name;
this.age=age;
this.getInfo=function(){
console.log(this.name+" is "+this.age)
}
} var Bob = new Tom("Bob",45); Bob.getInfo(); //Bob is 45
 

我们来点进阶的东西!!!

下面例子要解释的通 则需要两个重要的概念

延迟执行!

延迟调用!

8.将对象赋值给变量后,再调用方法,执行的对象仍然是Tom

var name="window";
var Tom={
name:"Tom",
show:function(){console.log(this.name)},
wait:function(){
var that=this;
that.show();
}
}; Tom.wait(); //Tom

因为show绑定在that的对象里,而that的值是对象Tom,所以执行的对象指向了Tom

"将对象保存到变量,再去调用方法",这个过程称为延迟调用,若不明白没关系

接着往下看

9.在函数上下文中若没有指明执行方法的对象,那么会由全局对象window来执行

var name="window";
var Tom={
name:"Tom",
show:function(){console.log(this.name)},
wait:function(){
var fun=this.show;
fun();
}
}; Tom.wait(); //window

fun赋值的是方法的内容,没有绑定于Tom对象里,因此执行的对象是window

上面的"把方法赋值给变量,再调用"这个过程可以看做延迟执行

延迟执行方法会导致该方法失去对象的绑定,导致对象由window执行执行

下面是一些延迟调用的例子

10.匿名函数的延迟

var name="window";
var Tom={
name:"Tom",
show:function(){console.log(this.name)},
wait:function(){!function(call){call();}(this.show)}
} Tom.wait(); //Window

11.setTimeout、setInterval函数延迟
这里只以setTimeout为例子

var name="window";
var Tom={
name:"Tom",
show:function(){console.log(this.name)},
wait:function(){setTimeout(this.show,1000)}
} Tom.wait(); //window

从上面两个例子可以得出:延迟执行会使得执行方法的对象更改为全局对象window

那么如何防止这种更改呢,我们尝试下面一招

12. 在延迟的环境下 尝试让Tom加班(对象也跟着延迟)

var name="window";
var Tom={
name:"Tom",
show:function(){console.log(this.name)},
wait:function(){setTimeout(Tom.show,1000)}
} Tom.wait(); //window

上面中this就直接指明Tom,尝试让Tom加班,但是结果仍然为Window对象

因为Tom.show放在第一个参数里,就相当于这个方法被保存到一个变量里

然后经过延迟后再执行这个变量,而保存的时候依然没有绑定到对象Tom,因此执行的对象依然是window

这个和上面第9个例子是一个道理的。称为延迟执行,只不过这次的延迟执行是用了setTimeout来实现

那么难道没有办法避免延迟执行方法导致执行对象被更改了吗?非也!

让对象也跟着延迟就可以做到!如下

13.虽然延迟会导致方法的执行对象被更改为Window 但也有办法防止执行对象更改 如下

var name="window"
var Tom ={
name : "Tom",
show : function(){console.log(this.name);},
wait: function(){
var that=this;
setTimeout(function(){that.show()},1000)}
} Tom.wait(); //Tom

通过变量保存对象,再使用变量去调用方法,达到执行的对象也跟着延迟的效果

14.eval函数的延迟

对于eval比较特殊

在eval环境下,执行的对象就是当前作用域的对象 如下

var name="window";
var Bob={
name:"Bob",
showName: function(){ eval("console.log(this.name)"); }
}; Bob.showName(); //Bob

15.eval函数的环境下,不会受到延迟而影响函数执行的对象

之所以eval特殊是因为eval不受延迟的影响

var name="window";
var that;
var Tom={
name:"Tom",
show:function(){console.log(this.name)},
wait:function(){that=this;setTimeout("that.show()",1000)}
} Tom.wait(); //Tom

也许你会觉得上面的代码没有eval函数的身影

其实setTimeout函数的第一个参数就是eval环境

他会指向当前执行作用域的执行对象,忽略延迟方法延迟调用

点击下面可以看看田小计划对JavaScript的this是如何见解

http://www.cnblogs.com/wilber2013/p/4909505.html

能把上面例子都理解了,那么this是一把有利的刀!

当然如果不能理解,那么像闭包一样 尽量的少用!

JavaScript中的this的更多相关文章

  1. javascript中的Array对象 —— 数组的合并、转换、迭代、排序、堆栈

    Array 是javascript中经常用到的数据类型.javascript 的数组其他语言中数组的最大的区别是其每个数组项都可以保存任何类型的数据.本文主要讨论javascript中数组的声明.转换 ...

  2. javascript中的this与函数讲解

    前言 javascript中没有块级作用域(es6以前),javascript中作用域分为函数作用域和全局作用域.并且,大家可以认为全局作用域其实就是Window函数的函数作用域,我们编写的js代码, ...

  3. JavaScript 中的数据类型

    Javascript中的数据类型有以下几种情况: 基本类型:string,number,boolean 特殊类型:undefined,null 引用类型:Object,Function,Date,Ar ...

  4. javascript中的操作符详解1

    好久没有写点什么了,根据博主的技术,仍然写一点javascript新手入门文章,接下来我们一起来探讨javascript的操作符. 一.前言 javascript中有许多操作符,但是许多初学者并不理解 ...

  5. 掌握javascript中的最基础数据结构-----数组

    这是一篇<数据结构与算法javascript描述>的读书笔记.主要梳理了关于数组的知识.部分内容及源码来自原作. 书中第一章介绍了如何配置javascript运行环境:javascript ...

  6. javascript中变量提升的理解

    网上找了两个经典的例子 var foo = 1; function bar() { if (!foo) { var foo = 10; } alert(foo); } bar(); // 10 var ...

  7. 前端开发:面向对象与javascript中的面向对象实现(二)构造函数与原型

    前端开发:面向对象与javascript中的面向对象实现(二)构造函数与原型 前言(题外话): 有人说拖延症是一个绝症,哎呀治不好了.先不说这是一个每个人都多多少少会有的,也不管它究竟对生活有多么大的 ...

  8. 简单分析JavaScript中的面向对象

    初学JavaScript的时候有人会认为JavaScript不是一门面向对象的语言,因为JS是没有类的概念的,但是这并不代表JavaScript没有对象的存在,而且JavaScript也提供了其它的方 ...

  9. Javascript中的valueOf与toString

    基本上,javascript中所有数据类型都拥有valueOf和toString这两个方法,null除外.它们俩解决javascript值运算与显示的问题,本文将详细介绍,有需要的朋友可以参考下. t ...

  10. 关于javascript中的this关键字

    this是非常强大的一个关键字,但是如果你不了解它,可能很难正确的使用它. 下面我解释一下如果在事件处理中使用this. 首先我们讨论一下下面这个函数中的this关联到什么. function doS ...

随机推荐

  1. css实现单行,多行文本溢出显示省略号……

    1.单行文本溢出显示省略号我们可以直接用text-overflow: ellipsis 实现方法: <style> .div_text{width: 300px; padding:10px ...

  2. SQL Server常见数据类型介绍

    数据表是由多个列组成,创建表时必须明确每个列的数据类型,以下列举SQL Server常见数据类型的使用规则,方便查阅. 1.整数类型 int 存储范围是-2,147,483,648到2,147,483 ...

  3. HTML块级元素

    前面的话   在HTML5出现之前,人们一般把元素分为块级.内联和内联块元素.本文将详细介绍HTML块级元素 h   标题(Heading)元素有六个不同的级别,<h1>是最高级的,而&l ...

  4. H5程序员如何利用cordova开发跨平台应用

    什么是Cordova? Cordova以前也叫PhoneGap,它提供了一组设备相关的API,通过这组API,移动应用能够以JavaScript访问原生的设备功能,如摄像头.麦克风等.Cordova还 ...

  5. Tomcat启动报错org.springframework.web.context.ContextLoaderListener类配置错误——SHH框架

    SHH框架工程,Tomcat启动报错org.springframework.web.context.ContextLoaderListener类配置错误 1.查看配置文件web.xml中是否配置.or ...

  6. PHP 设计模式概述

    一.设计模式(Design pattern)是什么? 设计模式是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了可重用代码.让代码更容易被他人理解.保证代码可靠性. ...

  7. 【干货分享】流程DEMO-事务呈批表

    流程名: 事务呈批表  业务描述: 办公采购.会议费用等事务的申请.流程发起时,会检查预算,如果预算不够,将不允许发起费用申请,如果预算够用,将发起流程,同时占用相应金额的预算,但撤销流程会释放相应金 ...

  8. windows 7(32/64位)GHO安装指南(U盘引导篇)~

    上一篇我们说了怎么制作U盘启动盘,那么这一篇让我们来看看如何进行正确的U盘引导启动. 现在的个人计算机一般分为台式机和笔记本,由于各厂商的喜好不同(开玩笑的啦),所以对于主板的BIOS设置各所不同.进 ...

  9. Xamarin中使用DatePickerDialog的相关问题

    在Xamarin中在使用Datepicker的时候,一般情况下只需要在对应的按钮或其他控件的点击事件中使用如下语句即可完成: EditText etBirthday = FindViewById< ...

  10. mono3.2.3+Jexus5.5+openSuSE13.1的asp.net

    读书的时候,我似乎有系统地学习过asp.net,但是基本已经还掉了...工作之后有做过一个内部用的网站,但也没有正式使用,的确只能算是个课程设计型的东西,不能做产品.后来工作需求是做Win8下的APP ...