针对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实现的类似百度搜索的输入框自动完成功能
Aug 23 Javascript
简介JavaScript中fixed()方法的使用
Jun 08 Javascript
jQuery实现复选框批量选择与反选的方法
Jun 17 Javascript
JS生成一维码(条形码)功能示例
Jan 19 Javascript
vue-router路由参数刷新消失的问题解决方法
Jun 17 Javascript
vue单页应用中如何使用jquery的方法示例
Jul 27 jQuery
在vue-cli搭建的项目中增加后台mock接口的方法
Apr 26 Javascript
基于vue中对鼠标划过事件的处理方式详解
Aug 22 Javascript
js jquery 获取某一元素到浏览器顶端的距离实现方法
Sep 05 jQuery
深入浅析javascript函数中with
Oct 28 Javascript
JavaScript中.min.js和.js文件的区别讲解
Feb 13 Javascript
对TypeScript库进行单元测试的方法
Jul 18 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中理解print EOT分界符和echo EOT的用法区别小结
2010/02/21 PHP
如何获知PHP程序占用多少内存(memory_get_usage)
2012/09/23 PHP
php性能分析之php-fpm慢执行日志slow log用法浅析
2016/10/17 PHP
php+js实现百度地图多点标注的方法
2016/11/30 PHP
Zend Framework前端控制器用法示例
2016/12/11 PHP
PHP实现打包zip并下载功能
2018/06/12 PHP
PDO::getAttribute讲解
2019/01/28 PHP
Checbox的操作含已选、未选及判断代码
2013/11/07 Javascript
jQuery中bind,live,delegate与one方法的用法及区别解析
2013/12/30 Javascript
Script标签与访问HTML页面详解
2014/01/10 Javascript
IE浏览器不支持getElementsByClassName的解决方法
2014/08/27 Javascript
js动态添加的DIV中的onclick事件简单实例
2016/07/25 Javascript
JavaScript的for循环中嵌套一个点击事件的问题解决
2017/03/03 Javascript
jQuery插件echarts实现的多柱子柱状图效果示例【附demo源码下载】
2017/03/04 Javascript
Vue中的scoped实现原理及穿透方法
2018/05/15 Javascript
JavaScript类的继承方法小结【组合继承分析】
2018/07/11 Javascript
AntV F2和vue-cli构建移动端可视化视图过程详解
2019/10/08 Javascript
js实现列表向上无限滚动
2020/01/13 Javascript
vue 在单页面应用里使用二级套嵌路由
2020/12/19 Vue.js
JavaScript中clientWidth,offsetWidth,scrollWidth的区别
2021/01/25 Javascript
Python 模块EasyGui详细介绍
2017/02/19 Python
python GUI实例学习
2017/11/21 Python
python实现计数排序与桶排序实例代码
2019/03/28 Python
python matplotlib折线图样式实现过程
2019/11/04 Python
OpenCV+Python--RGB转HSI的实现
2019/11/27 Python
在Python中利用pickle保存变量的实例
2019/12/30 Python
Pytorch使用MNIST数据集实现CGAN和生成指定的数字方式
2020/01/10 Python
django实现HttpResponse返回json数据为中文
2020/03/27 Python
python构造IP报文实例
2020/05/05 Python
利用Python实现学生信息管理系统的完整实例
2020/12/30 Python
CSS3 实现倒计时效果
2020/11/25 HTML / CSS
奠基仪式主持词
2014/03/20 职场文书
出纳担保书范文
2014/04/02 职场文书
祖国在我心中演讲稿600字
2014/05/04 职场文书
行政执法队伍作风整顿个人剖析材料
2014/10/11 职场文书
python flask框架快速入门
2021/05/14 Python