在这里必须要提一句的是,this指向是学习js必须要掌握的(必须),再开始之前先看底部的总结,然后回上面看例子便一目了然。

例子1:

function a(){
var user = "TangSir";
console.log(this.user); //undefined
console.log(this); //Window
}
a();
看总结第2条,这里函数本身没有被父级对象调用,那么这里就指向window
function a(){
var user = "TangSir";
console.log(this.user); //undefined
console.log(this);  //Window
}
window.a();
可以发现是一样的,这是因为window是全局对象,这里的a是被window点出来的,所以指向window(在这里也可以把window理解为a的父级,那么就是总结第3条)

例子2:
var o = {
user:"TangSir",
fn:function(){
console.log(this.user); //TangSir
}
}
o.fn();
这里的this指向o(总结第3条)
var o = {
user:"TangSir",
fn:function(){
console.log(this.user); //TangSir
}
}
window.o.fn();
这里加了个window,但是fn的父级还是o,是o调用了它(总结第3条)
var o = {
a:12,
b:{
a:14,
fn:function(){
console.log(this.a); //14
}
}
}
o.b.fn();
这里父级是b,那么this指向b(总结第3条)
例子3:
var o = {
a:12,
b:{
a:14,
fn:function(){
console.log(this.a); //undefined
console.log(this); //window
}
}
}
var j = o.b.fn;
j();
这里为何是undefined而不是14?是谁最后调用了它,这里是j(),那么this只能是window(总结第1条)
例子4:

function Fn(){
this.user = "TangSir";
}
var a = new Fn();
console.log(a.user); //TangSir
构造函数的this指向函数实例本身,这里赋给了a,等于赋值了一份给了a,那么a.user肯定也是TangSir(总结第5条)

例子5:
function Fn()
{
this.user = 'TangSir';
return {};
}
var a = new Fn();
console.log(a.user); //undefined
因为返回的是对象,所以this指向该对象,所有是undefined(总结第6条)
function Fn()
{
this.user = 'TangSir';
return function(){};
}
var a = new Fn();
console.log(a.user); //undefined
function同样是对象,所以this指向该对象,所有是undefined(总结第6条)
function Fn()
{
this.user = 'TangSir';
return 1;
}
var a = new Fn();
console.log(a.user); //TangSir
这里this指向函数实例本身,所以是TangSir(总结第6条)
function Fn()
{
this.user = 'TangSir';
return undefined;
}
var a = new Fn();
console.log(a.user); //TangSir
这里this指向函数实例本身,所以是TangSir(总结第6条)
function Fn()
{
this.user = 'TangSir';
return null;
}
var a = new Fn();
console.log(a.user); //TangSir
函数里有null比较特殊,这里this指向函数实例本身,所以是TangSir(总结第6条)
例子6:
var obj = {
name:"TangSir",
fn:function(){
fn1=()=>{
console.log(this.name); //TangSir
}
fn1();
}
}
obj.fn();

这里箭头函数父级是fn,去掉父级后,fn1()此时的父级就是obj,那么this.name自然是TangSir(总结第6条和第3条)

例子7:

var o = {
a:12,
b:{
a:14,
c:{
a:15,
d:function(){
console.log(this.a); //15
function e(){
console.log(this.a); //undefined
f();
}
e();
function f(){
console.log(this.a); //undefined
}
g=()=>{
console.log(this.a); //15
}
g();
}
}
}
}
o.b.c.d();
这里第一个console是15(总结4),第二和第三个console是undefined,不然理解,它们的调用时f()和e(),所以是undefined,第三个是箭头函数,根据例子6结合总结第7条,得出15

总结:

1、this在函数定义的时候是没办法确定指向的,只有函数执行的时候,最后谁调用了它才能确定this指向谁

2、如果函数中有this,但是函数本身没有被父级(上一级)对象调用,那么就指向window

3、如果函数中有this,且函数本身被父级(上一级)对象调用,那么this就指向上一级对象

4、如果函数中有this,且函数中有多个对象,尽管函数被最外层对象调用,但this仍然指向父级(上一级)对象

5、构造函数中this,指向构造函数实例,如果创建的实例赋给对象,那么等于复制了一份给对象,该对象也拥有实例中的this(new出来的构造函数可以改变this指向)

6、构造函数中带return,返回值若是对象,this指向的是那个返回的对象,返回值若是null,this还是指向那个构造函数实例

7、es6中=>箭头函数中的this,去掉当前函数的父级(上一级)对象,再看this指向谁,此时指向谁就是谁(结合第3条)

彻底搞懂js this指向问题的更多相关文章

  1. 帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)

    作为一名前端工程师,必须搞懂JS中的prototype.__proto__与constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞 ...

  2. 让你彻底搞懂JS中复杂运算符==

    让你彻底搞懂JS中复杂运算符== 大家知道,==是JavaScript中比较复杂的一个运算符.它的运算规则奇怪,容易让人犯错,从而成为JavaScript中“最糟糕的特性”之一. 在仔细阅读了ECMA ...

  3. 彻底搞懂 JS 中 this 机制

    彻底搞懂 JS 中 this 机制 摘要:本文属于原创,欢迎转载,转载请保留出处:https://github.com/jasonGeng88/blog 目录 this 是什么 this 的四种绑定规 ...

  4. 一文搞懂 js 中的各种 for 循环的不同之处

    一文搞懂 js 中的各种 for 循环的不同之处 See the Pen for...in vs for...of by xgqfrms (@xgqfrms) on CodePen. for &quo ...

  5. 彻底搞懂js __proto__ prototype constructor

    在开始之前,必须要知道的是:对象具有__proto__.constructor(函数也是对象固也具有以上)属性,而函数独有prototype 在博客园看到一张图分析到位很彻底,这里共享: 刚开始看这图 ...

  6. 通过一张简单的图,让你彻底地搞懂JS的==运算

    大家知道,JavaScript中的==是一种比较复杂运算,它的运算规则很奇怪,很容易让人犯错,从而成为JavaScript中“最糟糕的特性”之一. 在仔细阅读ECMAScript规范的基础上,我画了一 ...

  7. 晨叔技术晨报: 你真的搞懂JS中的“值传递”和“引用传递”吗?

    晨叔周刊,每周一话题,技术天天涨. 本周的话题是JS的内存问题(加入本周话题,请点击传送门). 图 话题入口 今天的技术晨报来,就来谈谈JS中变量的,值传递和引用传递的问题.现在,对于很多的JSer来 ...

  8. 一文彻底搞懂JS前端5大模块化规范及其区别

    码文不易,转载请带上本文链接,感谢~ https://www.cnblogs.com/echoyya/p/14577243.html 目录 码文不易,转载请带上本文链接,感谢~ https://www ...

  9. Spirit带你彻底搞懂JS的6种继承方案

    JavaScript中实现继承的6种方案 01-原型链的继承方案 function Person(){ this.name="czx"; } function Student(){ ...

随机推荐

  1. 说一说packet poll 错误掩码的一个bug tcp udp packet poll细节有所不同 处理时需要注意

    今天处理一个cpu标高的bug,原因:在poll 返回后将error事件当做POLLIN事件处理,fd 一直都在唤醒线程处理,但是rcv的时候没有数据: unsigned int datagram_p ...

  2. Adaboost算法的一个简单实现——基于《统计学习方法(李航)》第八章

    最近阅读了李航的<统计学习方法(第二版)>,对AdaBoost算法进行了学习. 在第八章的8.1.3小节中,举了一个具体的算法计算实例.美中不足的是书上只给出了数值解,这里用代码将它实现一 ...

  3. Vmware Tools is currently being installed on your system

    问题描述: 使用虚拟机安装Ubuntu过程中一直停留在"PLEASE WAIT! Vmware Tools is currently being installed on your syst ...

  4. Monitor的扩展支持string的超时锁

    对Monitor的使用可以防止lock的时间过长并且可以设置其对应的超时时间达到对预期代码的一个控制,合理的使用timeout可以有助于程序的健壮性.但是对于不同的并发程序可能某些时候我们需要的粒度是 ...

  5. 关于steam平台“wallpaper engine”软件出现界面黑屏,但壁纸能播放的问题

    前阵子重装电脑后,在使用wallpaper engine这款软件时发现了以下令人疑惑的画面: 点击"设置"和"壁纸选择"界面全是黑的......这还没完,更气人 ...

  6. vue项目中echarts属性总结

    <div id="echarts" style="width: 600px;height: 400px;margin-top: 100px;margin-left: ...

  7. 面试阿里,首先要掌握的 Java 泛型,帮你一次性搞懂!

    引言 泛型是Java中一个非常重要的知识点,在Java集合类框架中泛型被广泛应用.本文我们将从零开始来看一下Java泛型的设计,将会涉及到通配符处理,以及让人苦恼的类型擦除. 泛型基础 泛型类 我们首 ...

  8. 深度分析:Java 静态方法/变量,非静态方法/变量的区别,今天一并帮你解决!

    静态/非静态 方法/变量的写法 大家应该都明白静态方法/字段比普通方法/字段的写法要多一个static关键字,简单写下他们的写法吧,了解的可以直接略过 class Test{ // 静态变量 publ ...

  9. 在FL Studio中如何更好地为人声加上混响(进阶教程)

    为人声加上混响是我们在处理人声过程中必不可少的一步.然而,除了直接在人声混音轨道加上混响插件进行调节以外,这里还有更为细节的做法可以达到更好的效果. 步骤一:使用均衡器 在为人声加上混响之前,我们应该 ...

  10. 详讲FL Studio通道设置菜单

    我们在FL Studio"通道设置按钮"上右击鼠标就会弹出一个设置菜单,它包含了通道操作的各种常用命令.下文小编将会为大家详细讲解这些命令的具体作用,一起来学习吧! 1.首先,我们 ...