深入浅出this关键字

与python等其他的编程语言相比,this 关键字在 JavaScript 中的表现有所不同,是否是严格模式也会影响this的绑定;这里面套路众多,一不小心就可能出错,而且this作为动态作用域的表亲,与JavaScript的词法作用域差别很大,因此很多人称它为JavaScript的一个大坑。(我也是这样认为的,2333)

在绝大多数情况下,函数的调用方式决定了this的值。this不能在执行期间被赋值,并且在每次函数被调用时this的值也可能会不同。在此我们主要讨论一下较为常见的几种调用方式和在ES6中的箭头函数中的this。

几种较为常见的调用方式:

1.在全局范围内使用this,它将会指向全局对象即window.

var a = 1;
console.log(this.a);  // 1

console.log(this === window); // true

this.b = "hello,world";
console.log(window.b);  // "hello,world"
console.log(b);         // "hello,world"

2.在函数中调用时,也会指向全局对象。而在node中则指向global:

var a = 1;
function fn(){
  var a = 2;
  console.log(a);
}
fn(); //1

//所以要注意的是:
//在浏览器中:
fn() === window;

//在Node中:
fn() === global;

要注意的是在ES5的严格模式下,不存在全局变量。this将保持他进入执行上下文时的值,这种情况下this将会是undefined:

"use strict"
function fn(a){
    console.log(this);
}
fn(1);   //undefined

fn() === undefined; // true

3.方法调用text.foo(); 在这个例子时this指向text对象。要注意的是这样的行为,根本不受函数定义方式或位置的影响。而且this 的绑定只受最靠近的成员引用的影响

var text = {
  num: 1,
  foo: function() {
    return this.num;
  }
};

console.log(text.foo()); // 1

fn = text.foo;
text.a = {b: fn, num: 2};
console.log(text.a.b()); // 2

4.当一个函数用作构造函数时,它的this就会被绑定到正在构造的新对象中:

function Person(age){
  this.age = age;
}

var jake = new Person(18);
jake.age; //18

5.当使用call或者apply方法时,函数内的this将会被显式设置为函数调用的第一个参数。

//对象也可以作为call和apply传入的第一个参数,且this会显式的设置为该对象
var obj = {a: 1};

// 这个属性是在global对象定义的。
var a = 2;

function fn() {
  // this的值取决于函数的调用方式
  return this.a;  
}

fn();          // 2
fn.call(obj);  // 1
fn.apply(obj); // 1

用 call 和 apply 函数的时候还要注意,如果传递给 this 的值不是一个对象,JavaScript 会尝试使用内部 ToObject 操作将其转换为对象。

6.调用obj.bind()时可以创建一个与obj有着相同函数体和作用域的函数,但this将永久地被绑定到了bind的第一个参数,无论这个函数是如何被调用的。

var obj = {name: 'Jake'};
function sayName(){
    console.log(this.name)
};
var fn = sayName.bind(obj);  
// 注意 这里 fn还是一个函数
//功能和 sayName 一模一样,区别只在于它里面的 this 是 obj
fn() // 输出: 'Jake'

7.原型链中的 this仍然指向调用它的对象,这很好理解:

var fn = {
  add : function(){ 
    return this.a + this.b; 
  }
};
var p = Object.create(fn);
p.a = 1;
p.b = 2;

console.log(p.add()); // 3

8.当函数被用作事件处理函数时,它的this指向触发事件的元素:

  // 被调用时,将关联的元素变成蓝色
    function bluify(e){
      //在控制台打印出所点击元素
      console.log(this);
      //阻止时间冒泡
      e.stopPropagation();
      //阻止元素的默认事件
      e.preventDefault();      
      this.style.backgroundColor = '#A5D9F3';
    }
    // 获取文档中的所有元素的列表
    var elements = document.getElementsByTagName('*');

    // 将bluify作为元素的点击监听函数,当元素被点击时,就会变成蓝色
    for(var i=0 ; i<elements.length ; i++){
      elements[i].addEventListener('click', bluify, false);
    }

箭头函数

由于this的高复杂性,this绑定也成为了JavaScript中最常出错的因素之一。因此在ES6中的箭头函数没有了this绑定,必须通过查找作用域链来决定其值,而在全局代码中,它将被设置为全局对象:

var obj = this;
var foo = (() => this);
console.log(foo() === obj); // true

由于箭头函数不绑定this,所以 call() / apply() / bind() 方法对于箭头函数来说只是传入参数,对它的 this毫无影响。

//接着上面的代码

var a = {foo: foo};
console.log(a.foo() === obj); // true
// 用call来绑定this
console.log(foo.call(a) === obj); // true
// 用call来绑定this
foo = foo.bind(a);
console.log(foo() === obj); // true

考虑到 this 是词法层面上的,严格模式中与 this 相关的规则都将被忽略。(可以忽略是否在严格模式下的影响)

var a = () => {'use strict'; return this};
var b = () => { return this};
console.log(1,a() === window);
console.log(2,a() === b());
//1 true
//2 true

而当箭头函数作为方法调用时,this会是怎样的呢?

var obj = {
  a: 1,
  b: () => console.log(this.a, this),
  c: function() {
    console.log( this.a, this)
  }
}
obj.b();  // undefined Window{postMessage: ƒ, blur: ƒ, focus: ƒ, close: ƒ, frames: Window,…}
obj.c();  // 1 {a: 1, b: ƒ, c: ƒ}

可以看到的是作为方法的箭头函数this指向全局对象(window),而普通函数中的this则指向调用它的对象。

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

推荐阅读更多精彩内容