针对JavaScript中this指向的简单理解


Posted in Javascript onAugust 26, 2016

首先必须要说的是,this的指向在函数定义的时候是确定不了的,只有函数执行的时候才能确定this到底指向谁,实际上this的最终指向的是那个调用它的对象(这句话有些问题,后面会解释为什么会有问题,虽然网上大部分的文章都是这样说的,虽然在很多情况下那样去理解不会出什么问题,但是实际上那样理解是不准确的,所以在你理解this的时候会有种琢磨不透的感觉),那么接下来我会深入的探讨这个问题。 

为什么要学习this?如果你学过函数式编程,面向对象编程,那你肯定知道干什么用的,如果你没有学过,那么暂时可以不用看这篇文章,当然如果你有兴趣也可以看看,毕竟这是js中必须要掌握的东西。 

例子1:

function a(){
  var user = "小J";
  console.log(this.user); //undefined
  console.log(this); //Window
}
a();

按照我们上面说的this最终指向的是调用它的对象,这里的函数a实际是被Window对象所点出来的,下面的代码就可以证明。

function a(){
  var user = "小J";   
console.log(this.user); //undefined console.log(this);
} 
window.a() //Window

和上面代码一样吧,其实alert也是window的一个属性,也是window点出来的
例子2: 

var o = {
  user:"追梦子",
  fn:function(){
    console.log(this.user); //追梦子
  }
}
o.fn();

这里的this指向的是对象o,因为你调用这个fn是通过o.fn()执行的,那自然指向就是对象o,这里再次强调一点,this的指向在函数创建的时候是决定不了的,在调用的时候才能决定,谁调用的就指向谁,一定要搞清楚这个。 

其实例子1和例子2说的并不够准确,下面这个例子就可以推翻上面的理论。 

如果要彻底的搞懂this必须看接下来的几个例子.

例子3: 

var o ={ 
user:"追梦子", 
fn:function(){
 console.log(this.user); //追梦子
 } 

}
 window.o.fn();

这段代码和上面的那段代码几乎是一样的,但是这里的this为什么不是指向window,如果按照上面的理论,最终this指向的是调用它的对象,这里先说个而外话,window是js中的全局对象,我们创建的变量实际上是给window添加属性,所以这里可以用window点o对象。

这里先不解释为什么上面的那段代码this为什么没有指向window,我们再来看一段代码。 

var o = {
  a:10,
  b:{
    a:12,
    fn:function(){
      console.log(this.a); //12
    }
  }
}
o.b.fn();

这里同样也是对象o点出来的,但是同样this并没有执行它,那你肯定会说我一开始说的那些不就都是错误的吗?其实也不是,只是一开始说的不准确,接下来我将补充一句话,我相信你就可以彻底的理解this的指向的问题。 

================================================>>>>> 

情况1:如果一个函数中有this,但是它没有被上一级的对象所调用,那么this指向的就是window,这里需要说明的是在js的严格版中this指向的不是window,但是我们这里不探讨严格版的问题,你想了解可以自行上网查找。

================================================>>>>> 

情况2:如果一个函数中有this,这个函数有被上一级的对象所调用,那么this指向的就是上一级的对象。 

================================================>>>>> 

情况3:如果一个函数中有this,这个函数中包含多个对象,尽管这个函数是被最外层的对象所调用,this指向的也只是它上一级的对象,例子3可以证明,如果不相信,那么接下来我们继续看几个例子。 

var o = {
  a:10,
  b:{
    // a:12,
    fn:function(){
      console.log(this.a); //undefined
    }
  }
}
o.b.fn();

尽管对象b中没有属性a,这个this指向的也是对象b,因为this只会指向它的上一级对象,不管这个对象中有没有this要的东西。 

还有一种比较特殊的情况,例子4: 

var o = {
  a:10,
  b:{
    a:12,
    fn:function(){
      console.log(this.a); //undefined
      console.log(this); //window
    }
  }
}
var j = o.b.fn;
j();

这里this指向的是window,是不是有些蒙了?其实是因为你没有理解一句话,这句话同样至关重要。

this永远指向的是最后调用它的对象,也就是看它执行的时候是谁调用的,例子4中虽然函数fn是被对象b所引用,但是在将fn赋值给变量j的时候并没有执行所以最终指向的是window,这和例子3是不一样的,例子3是直接执行了fn。

this讲来讲去其实就是那么一回事,只不过在不同的情况下指向的会有些不同,上面的总结每个地方都有些小错误,也不能说是错误,而是在不同环境下情况就会有不同,所以我也没有办法一次解释清楚,只能你慢慢地的去体会。

构造函数版this: 

function Fn(){
  this.user = "小J";
}
var a = new Fn();
console.log(a.user); //小J

这里之所以对象a可以点出函数Fn里面的user是因为new关键字可以改变this的指向,将这个this指向对象a,为什么我说a是对象,因为用了new关键字就是创建一个对象实例,理解这句话可以想想我们的例子3,我们这里用变量a创建了一个Fn的实例(相当于复制了一份Fn到对象a里面),此时仅仅只是创建,并没有执行,而调用这个函数Fn的是对象a,那么this指向的自然是对象a,那么为什么对象Fn中会有user,因为你已经复制了一份Fn函数到对象a中,用了new关键字就等同于复制了一份。 

除了上面的这些以外,我们还可以自行改变this的指向 ========>>> call, apply, bind 

更新一个小问题当this碰到return时 

function fn() 
{ 
  this.user = '小J'; 
  return {}; 
}
var a = new fn; 
console.log(a.user); //undefined

再看一个 

function fn() 
{ 
  this.user = '小J'; 
  return function(){};
}
var a = new fn; 
console.log(a.user); //undefined

再来 

function fn() 
{ 
  this.user = '小J'; 
  return 1;
}
var a = new fn; 
console.log(a.user); //小J
function fn() 
{ 
  this.user = '小J'; 
  return undefined;
}
var a = new fn; 
console.log(a.user); //小J

如果返回值是一个对象,那么this指向的就是那个返回的对象,如果返回值不是一个对象那么this还是指向函数的实例。 

function fn() 
{ 
  this.user = '小J'; 
  return undefined;
}
var a = new fn; 
console.log(a); //fn {user: "小J"}

还有一点就是虽然null也是对象,但是在这里this还是指向那个函数的实例,因为null比较特殊。 

function fn() 
{ 
  this.user = '小J'; 
  return null;
}
var a = new fn; 
console.log(a.user); //小J

知识点补充: 
1.在严格版中的默认的this不再是window,而是undefined。 
2.new操作符会改变函数this的指向问题,虽然我们上面讲解过了,但是并没有深入的讨论这个问题,网上也很少说,所以在这里有必要说一下。 

function fn(){
  this.num = 1;
}
var a = new fn();
console.log(a.num); //1

为什么this会指向a?首先new关键字会创建一个空的对象,然后会自动调用一个函数apply方法,将this指向这个空对象,这样的话函数内部的this就会被这个空的对象替代。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jQuery动态地获取系统时间实现代码
May 24 Javascript
JS控制阿拉伯数字转为中文大写示例代码
Sep 04 Javascript
js字符串转换成数字与数字转换成字符串的实现方法
Jan 08 Javascript
动态创建script在IE中缓存js文件时导致编码的解决方法
May 04 Javascript
JavaScript实现仿新浪微博大厅和腾讯微博首页滚动特效源码
Sep 15 Javascript
JS控制层作圆周运动的方法
Jun 20 Javascript
jQuery简易时光轴实现方法示例
Mar 13 Javascript
Node.js 实现简单的接口服务器的实例代码
May 23 Javascript
AngularJS 仿微信图片手势缩放的实例
Sep 28 Javascript
用 Vue.js 递归组件实现可折叠的树形菜单(demo)
Dec 25 Javascript
JavaScript EventEmitter 背后的秘密 完整版
Mar 29 Javascript
AngularJS 事件发布机制
Aug 28 Javascript
轻松掌握JavaScript代理模式
Aug 26 #Javascript
轻松掌握JavaScript单例模式
Aug 25 #Javascript
很酷的星级评分系统原生JS实现
Aug 25 #Javascript
jQuery 利用$.ajax 时获取原生XMLHttpRequest 对象的方法
Aug 25 #Javascript
轻松掌握JavaScript策略模式
Aug 25 #Javascript
Javascript 6里的4个新语法
Aug 25 #Javascript
Javascript实现代码折叠功能
Aug 25 #Javascript
You might like
PHP中调用JAVA
2006/10/09 PHP
在任意字符集下正常显示网页的方法一
2007/04/01 PHP
php cache类代码(php数据缓存类)
2010/04/15 PHP
php学习笔记 [预定义数组(超全局数组)]
2011/06/09 PHP
Memcached常用命令以及使用说明详解
2013/06/27 PHP
ThinkPHP实现一键清除缓存方法
2014/06/26 PHP
PHP使用curl制作简易百度搜索
2016/11/03 PHP
yii框架使用分页的方法分析
2019/07/25 PHP
Html中JS脚本执行顺序简单举例说明
2010/06/19 Javascript
菜鸟javascript基础资料整理2
2010/12/06 Javascript
JavaScript获取/更改文本框的值的实例代码
2013/08/02 Javascript
JQuery判断子iframe何时加载完成解决方案
2013/08/20 Javascript
超级简单的jquery操作表格方法
2014/12/15 Javascript
javascript算法题:求任意一个1-9位不重复的N位数在该组合中的大小排列序号
2015/04/01 Javascript
javascript属性访问表达式用法分析
2015/04/25 Javascript
jquery实现浮动在网页右下角的彩票开奖公告窗口代码
2015/09/04 Javascript
JavaScript实现的多个图片广告交替显示效果代码
2015/09/04 Javascript
AngularJS教程之环境设置
2016/08/16 Javascript
详解Node 定时器
2018/02/26 Javascript
Vue 配合eiement动态路由,权限验证的方法
2018/09/26 Javascript
Vue3.0结合bootstrap创建多页面应用
2019/05/28 Javascript
vue轮播组件实现$children和$parent 附带好用的gif录制工具
2019/09/26 Javascript
解决vux 中popup 组件Mask 遮罩在最上层的问题
2020/11/03 Javascript
在Python中使用全局日志时需要注意的问题
2015/05/06 Python
使用Python的Scrapy框架十分钟爬取美女图
2016/12/26 Python
Python md5与sha1加密算法用法分析
2017/07/14 Python
pytorch实现对输入超过三通道的数据进行训练
2020/01/15 Python
python如何通过pyqt5实现进度条
2020/01/20 Python
美国汽车交易网站:Edmunds
2016/08/17 全球购物
美国隐形眼镜零售商:LensPure
2019/03/10 全球购物
世界上最大的乐谱选择:Sheet Music Plus
2020/01/18 全球购物
面试自我评价范文
2014/09/17 职场文书
基于Redis过期事件实现订单超时取消
2021/05/08 Redis
基于JavaScript实现年月日三级联动
2021/06/22 Javascript
Python Pandas数据分析之iloc和loc的用法详解
2021/11/11 Python
vue整合百度地图显示指定地点信息
2022/04/06 Vue.js