詳解JavaScript什么情況下不建議使用箭頭函數(shù)
箭頭函數(shù)作為ES6新增的語法,在使用時不僅能使得代碼更加簡潔,而且在某些場景避免this指向問題。但是箭頭函數(shù)不是萬能的,也有自己的缺點以及不適用的場景,雖然可以解決this只想問題,但是也可能會帶來this指向問題。具體場景具體分析,本文就深入探討箭頭函數(shù)。
箭頭函數(shù)沒有自己的this,其this取決于上下文中定義的this。
this指向原理
問題的由來
學懂 JavaScript 語言,一個標志就是理解下面兩種寫法,可能有不一樣的結(jié)果。
var obj = { foo: function () {} }; var foo = obj.foo; // 寫法一 obj.foo() // 寫法二 foo()
上面代碼中,雖然obj.foo
和foo
指向同一個函數(shù),但是執(zhí)行結(jié)果可能不一樣。請看下面的例子。
var obj = { foo: function () { console.log(this.bar) }, bar: 1 }; var foo = obj.foo; var bar = 2; obj.foo() // 1 foo() // 2
這種差異的原因,就在于函數(shù)體內(nèi)部使用了this
關(guān)鍵字。很多教科書會告訴你,this
指的是函數(shù)運行時所在的環(huán)境。對于obj.foo()
來說,foo
運行在obj
環(huán)境,所以this
指向obj
;對于foo()
來說,foo
運行在全局環(huán)境,所以this
指向全局環(huán)境。所以,兩者的運行結(jié)果不一樣。
這種解釋沒錯,但是教科書往往不告訴你,為什么會這樣?也就是說,函數(shù)的運行環(huán)境到底是怎么決定的?舉例來說,為什么obj.foo()
就是在obj
環(huán)境執(zhí)行,而一旦var foo = obj.foo
,foo()
就變成在全局環(huán)境執(zhí)行?
本文就來解釋 JavaScript 這樣處理的原理。理解了這一點,你就會徹底理解this
的作用。
內(nèi)存的數(shù)據(jù)結(jié)構(gòu)
JavaScript 語言之所以有this
的設(shè)計,跟內(nèi)存里面的數(shù)據(jù)結(jié)構(gòu)有關(guān)系。
var obj = { foo: 5 };
上面的代碼將一個對象賦值給變量obj
。JavaScript 引擎會先在內(nèi)存里面,生成一個對象{ foo: 5 }
,然后把這個對象的內(nèi)存地址賦值給變量obj
。
也就是說,變量obj
是一個地址(reference)。后面如果要讀取obj.foo
,引擎先從obj
拿到內(nèi)存地址,然后再從該地址讀出原始的對象,返回它的foo
屬性。
原始的對象以字典結(jié)構(gòu)保存,每一個屬性名都對應(yīng)一個屬性描述對象。舉例來說,上面例子的foo
屬性,實際上是以下面的形式保存的。
{ foo: { [[value]]: 5 [[writable]]: true [[enumerable]]: true [[configurable]]: true } }
注意,foo
屬性的值保存在屬性描述對象的value
屬性里面。
函數(shù)
這樣的結(jié)構(gòu)是很清晰的,問題在于屬性的值可能是一個函數(shù)。
var obj = { foo: function () {} };
這時,引擎會將函數(shù)單獨保存在內(nèi)存中,然后再將函數(shù)的地址賦值給foo
屬性的value
屬性。
{ foo: { [[value]]: 函數(shù)的地址 ... } }
由于函數(shù)是一個單獨的值,所以它可以在不同的環(huán)境(上下文)執(zhí)行。
var f = function () {}; var obj = { f: f }; // 單獨執(zhí)行 f() // obj 環(huán)境執(zhí)行 obj.f()
環(huán)境變量
JavaScript 允許在函數(shù)體內(nèi)部,引用當前環(huán)境的其他變量。
var f = function () { console.log(x); };
上面代碼中,函數(shù)體里面使用了變量x
。該變量由運行環(huán)境提供。
現(xiàn)在問題就來了,由于函數(shù)可以在不同的運行環(huán)境執(zhí)行,所以需要有一種機制,能夠在函數(shù)體內(nèi)部獲得當前的運行環(huán)境(context)。所以,this
就出現(xiàn)了,它的設(shè)計目的就是在函數(shù)體內(nèi)部,指代函數(shù)當前的運行環(huán)境。
var f = function () { console.log(this.x); }
上面代碼中,函數(shù)體里面的this.x
就是指當前運行環(huán)境的x
。
var f = function () { console.log(this.x); } var x = 1; var obj = { f: f, x: 2, }; // 單獨執(zhí)行 f() // 1 // obj 環(huán)境執(zhí)行 obj.f() // 2
上面代碼中,函數(shù)f
在全局環(huán)境執(zhí)行,this.x
指向全局環(huán)境的x
。
在obj
環(huán)境執(zhí)行,this.x
指向obj.x
。
回到本文開頭提出的問題,obj.foo()
是通過obj
找到foo
,所以就是在obj
環(huán)境執(zhí)行。一旦var foo = obj.foo
,變量foo
就直接指向函數(shù)本身,所以foo()
就變成在全局環(huán)境執(zhí)行。
箭頭函數(shù)的缺點
1.箭頭函數(shù)沒有arguments
參數(shù)列表,普通函數(shù)可以直接獲取到
arguments
是調(diào)用函數(shù)時,傳遞給函數(shù)的一個類似數(shù)組的對象,幾乎所有的函數(shù)都有此局部變量,可直接訪問并使用傳遞給函數(shù)的參數(shù)列表,箭頭函數(shù)除外。該變量不是數(shù)組對象,只是類似于數(shù)組,沒有數(shù)組的常用方法。
let fn1 = () => { console.log('arguments', arguments); } fn1(1, 2); // arguments is not defined let fn2 = function() { console.log('arguments', arguments); } fn2(1, 2); // Arguments對象,可查看具體的參數(shù)
2.無法通過apply、call、bind改變this的指向。箭頭函數(shù)的this默認指向父作用域或者當前調(diào)用對象,無法通過call等修改,但是function申明的函數(shù)可以修改this
this指向是js中經(jīng)常容易出錯的地方。箭頭函數(shù)的this指向是固定的,一般都是指向父作用域,默認指向window,不能在apply、call、bind中改變this的指向。普通函數(shù)的this指向不是固定的,有可能根據(jù)傳入的對象改變。
console.log('this1', this); // 指向window let fn3 = () => { console.log('this2', this); // 指向window } fn3.call({x: 'y'}); // 傳入新的對象 // fn3.apply({x: 'y'}); let fn4 = function() { console.log('this3', this); // 指向{x: 'y'} } fn4.call({x: 'y'});
不適用的場景
1.對象的方法,不建議使用箭頭函數(shù)
let obj = { key: 'key', getKey: () => { return this.key; }, getKey2() { return this.key; } }; obj.getKey(); // this指向window,返回值取決于window中是否有對應(yīng)的屬性 obj.getKey2(); // this指向obj,返回 'key'
2.對象的原型的方法,不建議使用箭頭函數(shù)
每個對象都有原型,原型也是一個對象,因此也不能添加箭頭函數(shù)的方法
let obj = { key: 'key' }; obj.__proto__.getKey = () => { console.log('this', this); // this指向window return this.key; } obj.getKey();
3.箭頭函數(shù)不能用作構(gòu)造函數(shù)
定義一個構(gòu)造函數(shù)可通過函數(shù)定義或者使用class定義一個類。箭頭函數(shù)不能用作構(gòu)造函數(shù),可使用普通函數(shù)
let fn5 = (userName, passwd) => { this.userName = userName; this.passwd = passwd; } let f1 = new fn5('張三', '123'); // fn5 is not a constructor console.log(f1.userName); let fn6 = function (userName, passwd) { this.userName = userName; this.passwd = passwd; } let f2 = new fn6('張三', '123'); console.log(f2.userName); // 張三
4.監(jiān)聽事件中需要使用this時不建議使用箭頭函數(shù)
比如在addEventListener
中,如果要在回調(diào)函數(shù)中使用this,那么就不建議使用箭頭函數(shù),而是應(yīng)該普通函數(shù),更好的是使用已定義的函數(shù)名,便于回收事件監(jiān)聽,避免可能的內(nèi)存泄漏。
dom.addEventListener('click', () => { console.log('this', this); // this指向window })
5.Vue的生命周期以及methods中的方法不建議使用箭頭函數(shù)
頁面中創(chuàng)建的Vue實例,本質(zhì)上來說也就是一個對象,其生命周期就是對應(yīng)的屬性,methods也是一個對象。在Vue的生命周期或者methods中使用箭頭函數(shù),則this的指向?qū)⒉皇钱斍癡ue實例,而是window對象,如果在方法中使用了this,則可能會拋出錯誤。
export default { mounted() {}, // mounted: () => {} methods: { getKey() {}, // getKey: () => {} } }
總結(jié)
- 箭頭函數(shù)有優(yōu)點,也有缺點,不可盲目使用,一定要清楚的知道為什么要使用箭頭函數(shù),為什么不能使用箭頭函數(shù)
- 箭頭函數(shù)可解決this指向,也可能帶來this指向問題
到此這篇關(guān)于詳解JavaScript什么情況下不建議使用箭頭函數(shù)的文章就介紹到這了,更多相關(guān)JavaScript箭頭函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
uni-app使用countdown插件實現(xiàn)倒計時
這篇文章主要為大家詳細介紹了uni-app使用countdown插件實現(xiàn)倒計時,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-11-11Django1.7+JQuery+Ajax驗證用戶注冊集成小例子
下面是散仙使用Django+Jquery+Ajax的方式來模擬實現(xiàn)了一個驗證用戶注冊時,用戶名存在不存在的一個小應(yīng)用。注意,驗證存在不存在使用的是Ajax的方式,不用讓用戶點擊按鈕驗證是否存在,需要的朋友可以參考下2017-04-04由JavaScript中call()方法引發(fā)的對面向?qū)ο罄^承機制call的思考
看到這里的call()方法,以前也看過手冊,說是對象冒充的,用于繼承的。在jQuery源碼里有點亂,所以就把這部分提取出來,放在一個單獨文件中,來看看具體執(zhí)行。2011-09-09移動端使用localStorage緩存Js和css文的方法(web開發(fā))
這篇文章主要介紹了web移動端使用localStorage緩存Js和css文的方法,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2016-09-09將字符串轉(zhuǎn)換成gb2312或者utf-8編碼的參數(shù)(js版)
直接在url中傳遞中文參數(shù)時,讀到的中文都是亂碼,那么我們應(yīng)該怎么將這些參數(shù)轉(zhuǎn)換呢,接下來與大家分享下將字符串轉(zhuǎn)換成utf-8或者gb2312編碼的參數(shù)的技巧2013-04-04Flow之一個新的Javascript靜態(tài)類型檢查器
今天我們興奮的發(fā)布了 Flow 的嘗鮮版,一個新的Javascript靜態(tài)類型檢查器。Flow為Javascript添加了靜態(tài)類型檢查,以提高開發(fā)效率和代碼質(zhì)量,本文給大家分享Flow之一個新的Javascript靜態(tài)類型檢查器,感興趣的朋友一起學習吧2015-12-12