欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解JavaScript什么情況下不建議使用箭頭函數(shù)

 更新時間:2022年06月24日 09:09:14   作者:諸葛小愚  
箭頭函數(shù)作為ES6新增的語法,在使用時不僅能使得代碼更加簡潔,而且在某些場景避免this指向問題。但是箭頭函數(shù)不是萬能的,也有自己的缺點以及不適用的場景,本文總結(jié)了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.foofoo指向同一個函數(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.foofoo()就變成在全局環(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)倒計時

    這篇文章主要為大家詳細介紹了uni-app使用countdown插件實現(xiàn)倒計時,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-11-11
  • JavaScript日期對象(Date)基本用法示例

    JavaScript日期對象(Date)基本用法示例

    這篇文章主要介紹了JavaScript日期對象(Date)基本用法,結(jié)合實例形式較為詳細的分析了JavaScript日期對象(Date)獲取日期、時間戳、年月日、星期及日期比對等操作技巧,需要的朋友可以參考下
    2017-01-01
  • 原生Js實現(xiàn)日歷掛件

    原生Js實現(xiàn)日歷掛件

    這篇文章主要為大家詳細介紹了原生Js實現(xiàn)日歷掛件,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-03-03
  • Django1.7+JQuery+Ajax驗證用戶注冊集成小例子

    Django1.7+JQuery+Ajax驗證用戶注冊集成小例子

    下面是散仙使用Django+Jquery+Ajax的方式來模擬實現(xiàn)了一個驗證用戶注冊時,用戶名存在不存在的一個小應(yīng)用。注意,驗證存在不存在使用的是Ajax的方式,不用讓用戶點擊按鈕驗證是否存在,需要的朋友可以參考下
    2017-04-04
  • 由JavaScript中call()方法引發(fā)的對面向?qū)ο罄^承機制call的思考

    由JavaScript中call()方法引發(fā)的對面向?qū)ο罄^承機制call的思考

    看到這里的call()方法,以前也看過手冊,說是對象冒充的,用于繼承的。在jQuery源碼里有點亂,所以就把這部分提取出來,放在一個單獨文件中,來看看具體執(zhí)行。
    2011-09-09
  • 移動端使用localStorage緩存Js和css文的方法(web開發(fā))

    移動端使用localStorage緩存Js和css文的方法(web開發(fā))

    這篇文章主要介紹了web移動端使用localStorage緩存Js和css文的方法,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-09-09
  • JavaScript設(shè)計模式之策略模式詳解

    JavaScript設(shè)計模式之策略模式詳解

    設(shè)計模式(Design pattern)是解決軟件開發(fā)某些特定問題而提出的一些解決方案也可以理解成解決問題的一些思路,下面這篇文章主要給大家介紹了關(guān)于JavaScript設(shè)計模式之策略模式的相關(guān)資料,需要的朋友可以參考下
    2022-06-06
  • 完美解決IE9瀏覽器出現(xiàn)的對象未定義問題

    完美解決IE9瀏覽器出現(xiàn)的對象未定義問題

    下面小編就為大家?guī)硪黄昝澜鉀QIE9瀏覽器出現(xiàn)的對象未定義問題。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧,祝大家游戲愉快哦
    2016-09-09
  • 將字符串轉(zhuǎn)換成gb2312或者utf-8編碼的參數(shù)(js版)

    將字符串轉(zhuǎn)換成gb2312或者utf-8編碼的參數(shù)(js版)

    直接在url中傳遞中文參數(shù)時,讀到的中文都是亂碼,那么我們應(yīng)該怎么將這些參數(shù)轉(zhuǎn)換呢,接下來與大家分享下將字符串轉(zhuǎn)換成utf-8或者gb2312編碼的參數(shù)的技巧
    2013-04-04
  • Flow之一個新的Javascript靜態(tài)類型檢查器

    Flow之一個新的Javascript靜態(tài)類型檢查器

    今天我們興奮的發(fā)布了 Flow 的嘗鮮版,一個新的Javascript靜態(tài)類型檢查器。Flow為Javascript添加了靜態(tài)類型檢查,以提高開發(fā)效率和代碼質(zhì)量,本文給大家分享Flow之一個新的Javascript靜態(tài)類型檢查器,感興趣的朋友一起學習吧
    2015-12-12

最新評論