最近在回顧js的一些基礎知識,把《你不知道的js》系列又看了一遍,this始終是重中之重,仍是決定把this相關知識作一個系統的總結,也方便本身往後回顧。面試
這是最經常使用的函數調用類型:獨立函數調用(即函數是直接使用不帶任何修飾的函數引用進行調用的)。能夠把這條規則看做是沒法應用其餘規則時的默認規則。
默認綁定的this在非嚴格模式下指向window,嚴格模式下指向undefined,好比下面的函數foo在非嚴格模式下:數組
var a = 2; function foo(){ var a = 3; console.log(this.a); } foo(); //2
這裏的foo()方法內的this指向了window,所以window.a = 2;安全
嚴格模式下,this.指向undefined,所以訪問this.a會報錯:app
var a = 2; function foo(){ "use strict"; var a = 3; console.log(this.a); } foo(); //Uncaught TypeError: Cannot read property 'a' of undefined
若是調用位置上有上下文對象,或者說被某個對象「擁有」或者「包
含」,則使用隱式綁定。函數
function foo() { console.log( this.a ); } var obj = { a: 2, foo: foo }; obj.foo(); // 2
上例中的foo是經過obj.foo()的方式調用的,調用位置會使用obj上下文來引用函數,所以foo中的this指向了obj。
另外foo是當作引用被加入到obj中的,可是不管是直接在obj 中定義仍是先定義再添加爲引用屬性,foo嚴格上來講都不屬於obj,所以上述定義裏面的「擁有」與「包含」加上了引號,這樣說是爲了方便理解。
常見的隱式調用場景:
obj.fn();
arguments[i]();
//其實就是將點的調用方式變爲了[]調用
el.onClick(function(){console.log(this);//this指向el})
this
先來看一段代碼:prototype
function foo() { console.log( this.a ); } var obj = { a: 2, foo: foo }; var bar = obj.foo; // 函數別名! var a = "global"; // a 是全局對象的屬性 bar(); // "global"
上述代碼其實只用看調用的方式:bar(),這實際上是一個不帶任何修飾的函數調用,所以應用了默認綁定。
還有一種參數傳遞的方式也會發生隱式丟失,原理其實跟上述例子同樣:code
function foo() { console.log( this.a ); } function doFoo(fn) { // fn 其實引用的是foo fn(); // <-- 調用位置! } var obj = { a: 2, foo: foo }; var a = "global"; // a 是全局對象的屬性 doFoo( obj.foo ); // "global"
使用call,apply和bind方法能夠指定綁定函數的this的值,這種綁定方法叫顯示綁定。對象
function foo() { console.log( this.a ); } var obj = { a:2 }; foo.call( obj ); // 2
經過foo.call(obj),咱們能夠在調用foo 時強制把它的this 綁定到obj 上繼承
new操做符能夠基於一個「構造函數」新建立一個對象實例,new的實例化過程以下:
function foo(a) { this.a = a; } var bar = new foo(2); console.log( bar.a ); // 2
new foo(2)後新建立了個實例對象bar,而後把這個新對象bar綁定到了foo函數中的this,所以執行this.a = a後實際上是把a賦給了bar.a
通常狀況下this的綁定會根據上述四條綁定規則來,那麼他們同時出現時,該以怎樣的順序來判斷this的指向?下面是具體的規則:
function foo() { console.log( this.a ); } var a = 2; foo.call( null ); // 2
什麼狀況下須要將上下文傳爲null呢?
1.使用bind函數來實現柯里化
function foo(a,b) { console.log(a,b); } // 使用 bind(..) 進行柯里化 var bar = foo.bind( null, 2 ); bar( 3 ); // 2,3
2.使用apply(..) 來展開一個數組,並看成參數傳入一個函數
function foo(a,b) { console.log(a,b); } // 把數組展開成參數 foo.apply( null, [2, 3] ); // 2,3
其實上面兩種使用場景其實都不關心call/app/bind第一個參數的值是什麼,只是想傳個佔位值而已。
可是老是傳入null可能會出現一些難以追蹤的bug,好比說當你在使用的第三方庫中的某個函數中有this時,this會被錯誤的綁定到全局對象上,形成一些難以預料的後果(修改全局變量)
var a = 1;//全局變量 const Utils = { a: 2, changeA: function(a){ this.a = a; } } Utils.changeA(3); Utils.a //3 a //1 Utils.changeA.call(null,4); Utils.a //3 a //4,修改了全局變量a!
更安全的作法:
var o = Object.create(null); Utils.changeA.call(o,6); a //1, 全局變量沒有修改 o.a // 6 改的是變量o
function foo() { console.log( this.a ); } var a = 2; var o = { a: 3, foo: foo }; var p = { a: 4 }; o.foo(); // 3 (p.foo = o.foo)(); // 2
賦值表達式p.foo = o.foo 的返回值是目標函數的引用,所以調用位置是foo() 而不是p.foo() 或者o.foo()。根據咱們以前說過的,這裏會應用默認綁定。
上述的幾種規則適用於全部的正常函數,但不包括ES6的箭頭函數。箭頭函數不使用this的四種標準規則,而是根據外層(函數或者全局)做用域(詞法做用域)來決定this
function foo() { // 返回一個箭頭函數 return (a) => { //this 繼承自foo() console.log( this.a ); }; } var obj1 = { a:2 }; var obj2 = { a:3 }; var bar = foo.call( obj1 ); bar.call( obj2 ); // 2, 不是3 !
foo() 內部建立的箭頭函數會捕獲調用時foo() 的this。因爲foo() 的this 綁定到obj1,bar(引用箭頭函數)的this 也會綁定到obj1,箭頭函數的綁定沒法被修改。(new 也不行!)
this的理論知識講解得差很少了,來幾個例子看看本身有沒有理解全面:
1.經典面試題:如下輸出結果是什麼
var length = 10; function fn() { console.log(this.length); } var obj = { length: 5, method: function(fn) { fn(); arguments[0](); } }; obj.method(fn, 1);
obj中method方法裏面調用了兩次fn。第一次是直接調用的「裸露」的fn,所以fn()中this使用默認綁定,this.length爲10.第二次調用時經過arguments0的方式調用的,arguments[0]其實指向的就是fn,可是是經過obj[fn]這種對象上下文的隱式綁定的,所以this指向arguments,而arguments只有一個一項(method中只有fn一個參數),所以arguments.length爲1。所以打印的結果爲:
10 1
2.如下輸出什麼
var obj = { birth: 1990, getAge: function () { var b = this.birth; // 1990 var fn = function () { return new Date().getFullYear() - this.birth; // this指向window或undefined }; return fn(); } }; obj.getAge();
答案是嚴格模式下會報錯
,非嚴格模式下輸出NaN
緣由也是由於在調用obj.getAge()後,getAge方法內的this使用隱式綁定。可是return fn()的時候用的是「裸露的fn」使用默認綁定,fn裏面的this指向window或者undefined。
使用箭頭函數來修正this的指向:
var obj = { birth: 1990, getAge: function () { var b = this.birth; // 1990 var fn = () => new Date().getFullYear() - this.birth; // this指向obj對象 return fn(); } }; obj.getAge(); // 25
使用箭頭函數後,fn中的this在他的詞法分析階段就已經肯定好了(即fn定義的時候),跟調用位置無關。fn的this指向外層的做用域(即getAge中的this)
3.如下輸出爲何是'luo'
var A = function( name ){ this.name = name; }; var B = function(){ A.apply(this,arguments); }; B.prototype.getName = function(){ return this.name; }; var b=new B('sven'); // B {name: "luo"} console.log( b.getName() ); // 輸出: 'luo'
執行new B('seven')後會返回一個新對象b,而且B函數中的this會綁定到新對象b上,B的函數體內執行A.apply(this.arguments)也就是執行b.name = name;這個時候b的值就是{name:'luo'},因此b.getName()就能輸出'luo'啦~
實際在業務使用中,邏輯會更復雜一些,可是萬變不離其宗,都按照上面寫的規則來代入就行了