JS基础系列——聊聊this

面向对象编程中,我们经常要和this打交道。而对于函数中this到底指向哪里,对刚开始接触JavaScript的我们,经常会分不清楚。

我们必须明确的是函数中this的指向不是固定的,和函数执行时的环境有关,简单的说,我们可以理解函数中this指向的是当前调用该函数的对象,所以我们也针对以下几种情况来消化下这种说话。

1、作为函数调用时

简单的例子如下:

function a(){
    console.log(this)
  b();
  function b(){
    console.log(this)
  }
}
a();

在浏览器时执行该代码时,打印出来的都是window对象。

因为作为函数调用时,默认是被认为在全局对象中调用该函数(nodejs环境中是global对象)注:不是在当前函数所处的作用域对象

这其实很难让人理解,为什么变成是全局对象,我们只能接受这种默认设定(也因为难以理解,所以现在严格模式下,默认是undefined),包括有时候我们将函数作为回调参数执行时声明的this也会是全局对象,如

setTimeout(function(){
    console.log(this)
},100)

有一点需要注意的是,箭头函数(我们等等再特殊讲下)

2、作为对象方法调用时

简单的例子如下

var obj={
    name:"obj",
        logThis(){
        console.log(this,this.name);
    }
}
obj.logThis();

代码执行时,答应出来的是obj对象,name值是obj,这是满足我们的预期:函数目前就是作为obj的方法被调用,所以调用该函数的对象是obj。

这种情况我们很容易消化,但是换了一种情况就另当别论,如

var name="test";
var obj={
    name:"obj",
        logThis(){
        console.log(this,this.name);
    }
}
var fn=obj.logThis;
fn();

这个时候打印出来的window对象,name值是test

原来这个时候fn指向的是一个函数(不是指向 obj执行logThis 这个行为),fn调用就是调用一个函数,也就是第一种情况,所以上面的例子就如下:

var name="test";
var obj={
    name:"obj",
        logThis(){
        console.log(this,this.name);
    }
}
var fn=function(){
        console.log(this,this.name);
}
fn();

3、作为构造函数执行

简单的例子如下

function People(){
console.log(this)
}
People.prototype={
name:"people"
};
new People();
这个时候打印出来的是我们新生成的people对象。

因为构造函数执行的过程我们可以简单理解为

1、新生成一个对象(people)

2、该对象原型对象指向构造函数的prototype对象(People.prototype)

3、通过新生成的对象调用构造函数(可以简单理解为People.call(this))

也就是在第三步的时候明确声明了当前调用函数的对象是新生成的对象

4、作为箭头函数调用

简单例子如下

var obj={
    logThis(){
        setTimeout(function(){
        console.log(this);
      })
        setTimeout(()=>{
        console.log(this)
      })
  }
}
obj.logThis();

这个时候打印的我们可能因为都是window对象,然而结果打脸了,第一个打印的是window对象,第二个打印的是obj对象

原来箭头函数,不会生成新的this指向,箭头函数在哪个环境执行,指向的就是当前环境的this对象,也就是第二种情况下,this指向的obj.logThis()执行时的this指向(前面我们提到的是obj对象)

5、apply,call,bind调用

this的指针默认指向就是上面提到的4种,但是总有需求,我们想要自定义this指向,这个时候apply,call,bind这3个方法就排的上用场

这三个方法都是函数对象自身的方法,第一个参数支持传入的是函数调用时设置的this对象,如

let obj={
    name:"obj"
}
function test(){
    console.log(this)
}

test();
test.call(obj);

这个时候第一个打印的是window对象(看1说明),第二个打印的是obj对象(我们通过传参的方法明确说明函数执行是的this对象是obj)

而这三个方法,不同如下

apply,call 调用时都是直接调用函数,并把第一个参数的函数作为函数的this指向
不同的是apply传递给执行函数的参数是数组格式,而call传递的参数直接附加即可
let obj={};
function test(a,b){
    console.log(this,a,b)
}
test.call(obj,1,2);
test.apply(obj,[1,2]);
  
bind调用时是返回明确好函数this指针的新函数,而不是立即执行,需要再次调用
var test2=test.apply(obj);
test2(1,2)

上述总结希望对大家有用

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,451评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,172评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,782评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,709评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,733评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,578评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,320评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,241评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,686评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,878评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,992评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,715评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,336评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,912评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,040评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,173评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,947评论 2 355

推荐阅读更多精彩内容