這篇文章主要講解了js this 綁定機制的用法,內(nèi)容清晰明了,對此有興趣的小伙伴可以學習一下,相信大家閱讀完之后會有幫助。
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)!專注于網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、微信小程序定制開發(fā)、集團企業(yè)網(wǎng)站建設(shè)等服務項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了蔡家坡免費建站歡迎大家使用!
函數(shù)調(diào)用位置
與詞法作用域相反的是,this的指向由函數(shù)運行時決定,它是動態(tài)的,隨著函數(shù)調(diào)用位置變化而變化。
要理解 this,首先要理解調(diào)用位置:調(diào)用位置就是函數(shù)在代碼中被調(diào)用的位置(而不是聲明的位置)。只有仔細分析調(diào)用位置才能回答這個問題:這個this到底引用的是什么?
function baz() { // 當前調(diào)用棧是:baz // 因此,當前調(diào)用位置是全局作用域 console.log( "baz" ); bar(); // <-- bar的調(diào)用位置 } function bar() { // 當前調(diào)用棧是baz -> bar // 因此,當前調(diào)用位置在baz中 console.log( "bar" ); foo(); // <-- foo的調(diào)用位置 } function foo() { // 當前調(diào)用棧是baz -> bar -> foo // 因此,當前調(diào)用位置在bar中 console.log( "foo" ); } baz(); // <-- baz的調(diào)用位置
多數(shù)現(xiàn)代桌面瀏覽器都內(nèi)置了開發(fā)者工具,其中包含JavaScript調(diào)試器。你可以在工具中給函數(shù)的第一行代碼設(shè)置一個斷點,或者直接在第一行代碼之前插入一條 debugger;語句。運行代碼時,調(diào)試器會在那個位置暫停,同時會展示當前位置的函數(shù)調(diào)用列表,這就是你的調(diào)用棧。因此,如果你想要分析this的綁定,使用開發(fā)者工具得到調(diào)用棧,然后找到棧中第二個元素,這就是真正的調(diào)用位置。
this 綁定規(guī)則
函數(shù)的調(diào)用位置決定了this的綁定對象,當我們找到調(diào)用位置后,然后判斷需要應用下面四條規(guī)則中的哪一條。
獨立函數(shù)調(diào)用
獨立函數(shù)調(diào)用,this 指向函數(shù)調(diào)用位置所在的包含環(huán)境對象。
function foo() { console.log( this.a ); } var a = 2; foo(); // 2
作為對象的方法調(diào)用
當函數(shù)作為某個對象的方法被調(diào)用時,this 指向這個對象。
function foo() { console.log( this.a ); } var obj = { a: 2, foo: foo }; obj.foo(); // 2
特別注意:雖然函數(shù)foo并不屬于obj對象,但調(diào)用位置使用obj的上下文來調(diào)用函數(shù)。我一直在強調(diào)調(diào)用位置的重要性,因為你可能一不留神就會忽略掉它,看下面的列子:
function foo() { console.log( this.a ); } var obj = { a: 2, foo: foo }; var bar = obj.foo; // 函數(shù)別名! 步驟1 var a = "oops, global"; // a是全局對象的屬性 bar(); // "oops, global" 步驟2
在步驟1中,變量bar是obj.foo 的一個引用,它實際指向的是函數(shù)foo。所以使用bar()與直接使用foo()并沒有不同。
使用 .call/ .apply 綁定
每創(chuàng)建一個函數(shù),這個函數(shù)就有了兩個繼承而來的方法:call和apply。
它們的第一個參數(shù)是一個對象,它們會把這個對象綁定到this,接著在調(diào)用函數(shù)時指定這個 this 。因為你可以直接指定 this 的綁定對象,因此我們稱之為顯式綁定。
function foo() { console.log( this.a ); } var obj = { a:2 }; foo.call( obj ); // 2
new綁定
使用 new 來調(diào)用函數(shù),或者說發(fā)生構(gòu)造函數(shù)調(diào)用時,會自動執(zhí)行下面的操作。
function foo(a) { this.a = a; } var bar = new foo(2); console.log( bar.a ); // 2
使用 new 來調(diào)用 foo(..) 時,我們會構(gòu)造一個新對象并把它綁定到 foo(..) 調(diào)用中的 this 上。
優(yōu)先級
如果要判斷一個運行中函數(shù)的this綁定,就需要找到這個函數(shù)的直接調(diào)用位置。找到之后就可以順序應用下面這四條規(guī)則來判斷 this 的綁定對象。
一定要注意,有些調(diào)用可能在無意中使用默認綁定規(guī)則。如果想“更安全”地忽略 this 綁定,你可以使用一個DMZ對象,比如 ø = Object.create(null) ,以保護全局對象。
看完上述內(nèi)容,是不是對js this 綁定機制的用法有進一步的了解,如果還想學習更多內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。