細(xì)說JavaScript中的this指向與綁定規(guī)則
this的指向
通常在瀏覽器中,在全局中使用this,該this指向的是window對象,無論是否開啟了嚴(yán)格模式。但我們一般使用this,也是在函數(shù)中進(jìn)行使用。所有的函數(shù)在被調(diào)用的時候,會產(chǎn)生一個函數(shù)執(zhí)行上下文,該上下文中記錄這函數(shù)的作用域鏈、AO對象以及當(dāng)前這個函數(shù)的this指向。我們看下面代碼:
function foo() { console.log(this) } foo()//window var obj = { name: 'why', foo: foo } obj.foo()//obj foo.apply("abc")//String{'abc'}
上面代碼中,我們定義了一個函數(shù)foo來輸出調(diào)用它的時候this的指向,從上往下分別指向的是window、obj、String{'abc'},所以我們可以知道:
- 函數(shù)在調(diào)用時,JavaScript會默認(rèn)給this綁定一個值;
- this的綁定和定義的位置(編寫的位置)沒有關(guān)系;
- this的綁定和調(diào)用方式以及調(diào)用的位置有關(guān)系;
- this是在運(yùn)行時被動態(tài)綁定的,而不是編譯時固定下來的
判斷this綁定的4條規(guī)則
規(guī)則1:默認(rèn)綁定
獨(dú)立函數(shù)調(diào)用的時候,this永遠(yuǎn)綁定的是window對象,如下示例:
function foo() { console.log(this) } function foo2() { foo() } var obj = { name: "xiaomin", foo: function() { console.log(this) } } foo()//window foo2()//window var bar = obj.foo bar()//window
規(guī)則2:隱式綁定
通過某個對象發(fā)起調(diào)用的函數(shù),哪個對象發(fā)起調(diào)用的函數(shù),那么該調(diào)用函數(shù)的this就隱式綁定在了這個對象:
var obj1 = { name: "obj1", foo: function () { console.log(this); }, }; var obj2 = { name: "obj2", bar: obj1.foo, obj1: obj1, }; obj1.foo();//obj1 obj2.obj1.foo();//obj1 obj2.bar();//obj2
規(guī)則3:顯示綁定
通過函數(shù)的方法call、apply、bind將this顯示綁定了某個對象上,那么該函數(shù)的this就指向到這個對象上:
function foo() { console.log(this) } var obj = { name: "obj" } foo.call(obj) foo()//obj foo.apply(obj) foo()//obj var newfoo = foo.bind(obj) newfoo()//obj
規(guī)則4:new綁定
我們有如下代碼:
function Person(name, age) { this.name = name this.age = age console.log(this)//p } var p = new Person("kobe",12);
在這里我們有一個構(gòu)造函數(shù)Person,并用它創(chuàng)建了一個對象,這里簡單說明一下通過new調(diào)用構(gòu)造函數(shù)創(chuàng)建對象時的5個步驟:
- 在內(nèi)存中創(chuàng)建一個空對象
- 將該對象的[[prototype]]指向構(gòu)造函數(shù)的prototype屬性
- 構(gòu)造函數(shù)里的this指向這個空對象
- 開始執(zhí)行構(gòu)造函數(shù)的函數(shù)體代碼
- 如果構(gòu)造函數(shù)未返回內(nèi)容,將該對象作為返回值返回出去
從上面我們可以看到,在使用new創(chuàng)建對象的時候,在第三個步驟會將構(gòu)造函數(shù)內(nèi)部的this綁定到創(chuàng)建的這個對象上,也就是上面代碼中,Person函數(shù)中打印的this就是會是通過new關(guān)鍵字創(chuàng)建的對象p。
this綁定的優(yōu)先級
當(dāng)同時出現(xiàn)上列規(guī)則的多種規(guī)則綁定的時候,這時候this的指向就會根據(jù)上面綁定規(guī)則的優(yōu)先級進(jìn)行綁定:
- 默認(rèn)綁定優(yōu)先級最低
- 顯示綁定優(yōu)先于隱式綁定,如下列代碼打印的this就不是obj而是String {'aaa'}
function foo() { console.log(this);//String?{'aaa'} } var obj = { name: "obj", foo: foo.bind("aaa"), }; obj.foo();
- new綁定優(yōu)先于隱式綁定,如下列代碼打印出來的就不是obj,而是創(chuàng)建出來的對象f
var obj = { name: "obj", foo: function () { this.name = "kkk"; console.log("this:", this);//foo?{name: 'kkk'} }, }; var f = new obj.foo();
- new綁定優(yōu)先bind綁定(new綁定不能同時與call、apply同時使用),如下代碼this打印出來的就是obj的對象而不是String{'aaa'}
function foo() { console.log(this); } var bar = foo.bind("aaa"); var obj = new bar(); //var obj2 = new foo.apply("123");//這樣寫會報錯
除了四個this綁定規(guī)則外的特殊情況
1、在顯示綁定中我們綁定了null或者undefined,this會綁定到window上
function foo() { console.log(this) } foo.call({})//{} foo.call(null)//window foo.call(undefined)//window
2、函數(shù)的間距引用,在下列代碼中,賦值表達(dá)式 p.foo = o.foo 的返回值是目標(biāo)函數(shù)的引用,因此調(diào)用位置是 foo(),不是o.foo()。所以打印出來的this是window而不是o或者p
function foo() { console.log(this); } var o = { foo: foo }; var p = {}; o.foo(); // o (p.foo = o.foo)(); // window
3、箭頭函數(shù)。箭頭函數(shù)是不會有自己的this的,在調(diào)用箭頭函數(shù)的時候,在它的函數(shù)執(zhí)行上下文中并不會生成它this的記錄,而如果在箭頭函數(shù)的函數(shù)體中使用到了this,那么它將會在它的上層作用域里面去找this,看下面代碼
var obj = { data: [], changeData: function () { console.log("普通函數(shù)this:", this); (() => { console.log("箭頭函數(shù)this:", this); })(); setTimeout(() => { this.data = ["1", "2", "3"]; console.log("this:", this); }, 1000); }, }; obj.changeData();
打印結(jié)果:
可以看出,上面打印的this其實(shí)都是指向的obj。在箭頭函數(shù)內(nèi)它自身沒有this,那么它就向外層尋找,找到了changeData這個函數(shù),而這時由于隱式綁定規(guī)則,changeData由obj調(diào)用的,那么changeData的this就指向的obj,所以箭頭函數(shù)的this指向的就是obj。
以上就是細(xì)說JavaScript中的this指向與綁定規(guī)則的詳細(xì)內(nèi)容,更多關(guān)于JavaScript this的資料請關(guān)注腳本之家其它相關(guān)文章!
- 一文全面解析JS中的this綁定規(guī)則
- JavaScript中this綁定規(guī)則你理解了嗎
- JavaScript this綁定與this指向問題的解析
- JavaScript?中的?this?綁定規(guī)則詳解
- JavaScript中this的綁定你知道幾種?
- 詳解JavaScript中的this硬綁定
- 一文搞懂JavaScript中的this綁定規(guī)則
- JavaScript中?this?的綁定指向規(guī)則
- 詳解JavaScript的this指向和綁定
- JavaScript this綁定過程深入詳解
- React.js綁定this的5種方法(小結(jié))
- JavaScript調(diào)用模式與this關(guān)鍵字綁定的關(guān)系
- 深入理解JavaScript this綁定規(guī)則
相關(guān)文章
JavaScript 文件加載與阻塞問題之性能優(yōu)化案例詳解
這篇文章主要介紹了JavaScript 文件加載與阻塞問題之性能優(yōu)化案例詳解,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09用javascript實(shí)現(xiàn)自動輸出網(wǎng)頁文本
這篇文章主要介紹了用javascript實(shí)現(xiàn)自動輸出網(wǎng)頁文本,用到兩個函數(shù):setTimeout(),遞歸和String.substring();,需要的朋友可以參考下2015-07-07JS/jQuery實(shí)現(xiàn)獲取時間的方法及常用類完整示例
這篇文章主要介紹了JS/jQuery實(shí)現(xiàn)獲取時間的方法及常用類,結(jié)合完整實(shí)例形式分析了javascript針對日期時間的獲取、轉(zhuǎn)換、計(jì)算與檢測相關(guān)操作技巧,需要的朋友可以參考下2019-03-03淺析JavaScrip哪些操作會造成內(nèi)存泄露以及預(yù)防方法
在?JavaScript?中,內(nèi)存泄露是指程序不再使用的內(nèi)存沒有被釋放,從而導(dǎo)致內(nèi)存的持續(xù)增長,最終可能導(dǎo)致性能下降或應(yīng)用崩潰,本文整理了一些容易造成內(nèi)存泄漏的操作以及預(yù)防方法,需要的可以了解下2024-12-12JS使用setInterval實(shí)現(xiàn)的簡單計(jì)時器功能示例
這篇文章主要介紹了JS使用setInterval實(shí)現(xiàn)的簡單計(jì)時器功能,涉及javascript基于setInterval的定時觸發(fā)與數(shù)值運(yùn)算相關(guān)操作技巧,需要的朋友可以參考下2018-04-04JS驗(yàn)證input輸入框(字母,數(shù)字,符號,中文)
本文主要介紹了JS驗(yàn)證input輸入框(字母,數(shù)字,符號,中文)的方法。具有很好的參考價值。下面跟著小編一起來看下吧2017-03-03JS/jQuery實(shí)現(xiàn)默認(rèn)顯示部分文字點(diǎn)擊按鈕顯示全部內(nèi)容
默認(rèn)顯示部分文字,點(diǎn)擊按鈕顯示全部,類似這樣的功能在一些特殊的地方會見到吧,下面與大家分享下JS、jQuery如何實(shí)現(xiàn),感興趣的朋友可以參考下哈,希望對你有所幫助2013-05-05