自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

JavaScript中閉包的四個(gè)有用技巧

開(kāi)發(fā) 前端
在本文中,我們深入探討了JavaScript中閉包的4種有用技巧,以及如何應(yīng)用它們來(lái)解決各種問(wèn)題和提高代碼質(zhì)量。這些技巧包括解決循環(huán)中的變量作用域問(wèn)題,實(shí)現(xiàn)記憶功能以提高性能,封裝私有變量和屬性,以及使用函數(shù)柯里化來(lái)提高函數(shù)的靈活性。

本文轉(zhuǎn)載自微信公眾號(hào)「黑土豆的前端博客」,作者M(jìn)aybe007。轉(zhuǎn)載本文請(qǐng)聯(lián)系黑土豆的前端博客公眾號(hào)。

前言

當(dāng)談到JavaScript編程中的高級(jí)概念和技巧時(shí),閉包(Closures)是一個(gè)重要而有趣的主題。閉包是一種函數(shù)與其創(chuàng)建時(shí)的詞法環(huán)境的組合,它允許我們捕獲和保留局部變量,并在函數(shù)之外使用它們。在這篇文章中,我們將深入探討JavaScript中閉包的4種有用技巧,以及如何應(yīng)用它們來(lái)解決各種問(wèn)題和提高代碼質(zhì)量。

1. 解決循環(huán)中的問(wèn)題

在JavaScript中,循環(huán)中的變量作用域問(wèn)題經(jīng)常會(huì)導(dǎo)致預(yù)期之外的結(jié)果。通常,使用var聲明變量會(huì)導(dǎo)致循環(huán)中的變量共享相同的作用域,因此在異步操作中,這些變量可能會(huì)具有意外的值。

問(wèn)題場(chǎng)景:

for (var i = 0; i < 3; i++) {
    setTimeout(() => {
        console.log(i); // 輸出什么?
    }, 1000 * i);
}

上述代碼將在1秒、2秒和3秒后分別打印出3,三次都是相同的值。這是因?yàn)閟etTimeout是異步的,它在循環(huán)結(jié)束后才執(zhí)行,此時(shí)i的值已經(jīng)是3。

解決方法:

使用閉包來(lái)保存每次迭代中的i的值:

for (var i = 0; i < 3; i++) {
    ((n) => {
        setTimeout(() => {
            console.log(n); // 輸出0、1、2
        }, 1000 * n);
    })(i);
}

或者,更簡(jiǎn)單的方式是使用let來(lái)聲明循環(huán)變量,它將在每次迭代中創(chuàng)建一個(gè)新的作用域:

for (let i = 0; i < 3; i++) {
    setTimeout(() => {
        console.log(i); // 輸出0、1、2
    }, 1000 * i);
}

這2種方法都可以解決循環(huán)中的作用域問(wèn)題。

2. 保存函數(shù)狀態(tài)

閉包還可以用于實(shí)現(xiàn)記憶功能,通過(guò)緩存計(jì)算結(jié)果來(lái)提高性能。這在需要重復(fù)計(jì)算的函數(shù)中特別有用。

問(wèn)題場(chǎng)景:

實(shí)現(xiàn)一個(gè)累加器:

let sum = 1;


function add(num) {
    sum += num;
    return sum;
}


console.log(add(1));
console.log(add(5));

每次調(diào)用add時(shí),它都會(huì)將上次的值保存下來(lái)。但是這段代碼有潛在的問(wèn)題,那就是sum可能會(huì)被其他部分代碼無(wú)意中修改。那如何解決?

解決方法:

使用閉包就可以規(guī)避上面存在的問(wèn)題且可以緩存已經(jīng)計(jì)算的值:

function calculator(val) {
    let sum = val;
    return function(num) {
        sum += num;
        return sum;
    }
}
const add = calculator(1);
console.log(add(1)); //2
console.log(add(5)); //7

每次調(diào)用返回的函數(shù)時(shí),它都會(huì)將傳遞給它的數(shù)字加到總和中,并返回新的總和。

3. 封裝私有變量和屬性

在過(guò)去,為了保護(hù)對(duì)象的私有變量,常常使用閉包。通過(guò)閉包,可以將變量封裝在函數(shù)內(nèi)部,只能通過(guò)函數(shù)暴露的接口來(lái)訪問(wèn)和修改。

問(wèn)題場(chǎng)景:

function add() {
    let count = 0;
    count++;
    console.log(count);
}
add(); //輸出1
add(); //輸出1
add(); //輸出1

調(diào)用函數(shù),輸出的結(jié)果都是1,但是顯然我們想要的效果是讓count每次加1的。那如何解決呢?

解決方法:

使用閉包來(lái)封裝私有變量:

function add(){
    let count = 0;
    function a() {
        count++;
        console.log(count);
    }
    return a;
}
var res = add();
res() //1
res() //2
res() //3

add函數(shù)返回了一個(gè)閉包a,其中包含了count變量。由于count只在add函數(shù)內(nèi)部定義,因此外部無(wú)法直接訪問(wèn)它。但是,由于a函數(shù)引用了count變量,因此count變量的值可以在閉包內(nèi)部被修改和訪問(wèn),這樣就可以防止它被惡意修改了。

4. 函數(shù)柯里化

函數(shù)柯里化是一種將接受多個(gè)參數(shù)的函數(shù)轉(zhuǎn)化為一系列接受一個(gè)參數(shù)的函數(shù)的過(guò)程。這也可以通過(guò)閉包來(lái)實(shí)現(xiàn)。

問(wèn)題場(chǎng)景:

const add = (a, b, c) => {
    return a + b + c;
}


console.log(add(2, 3, 4)); // 輸出9

解決方法:

使用閉包來(lái)實(shí)現(xiàn)函數(shù)柯里化:

function curry(callback) {
    const args = [];


    return function curried(...newArgs) {
        args.push(...newArgs);


        if (args.length >= callback.length) {
            return callback(...args);
        } else {
            return curried;
        }
    };
}


function add(a, b, c) {
    return a + b + c;
}


const curriedAdd = curry(add);


console.log(curriedAdd(2)(3)(4)); // 輸出 9

函數(shù)柯里化使函數(shù)更加靈活,能夠逐步接受參數(shù),提高代碼的可重用性和可讀

總結(jié)

在本文中,我們深入探討了JavaScript中閉包的4種有用技巧,以及如何應(yīng)用它們來(lái)解決各種問(wèn)題和提高代碼質(zhì)量。這些技巧包括解決循環(huán)中的變量作用域問(wèn)題,實(shí)現(xiàn)記憶功能以提高性能,封裝私有變量和屬性,以及使用函數(shù)柯里化來(lái)提高函數(shù)的靈活性。

閉包是JavaScript中一個(gè)強(qiáng)大的概念,它允許我們?cè)诤瘮?shù)之外訪問(wèn)和操作局部變量,從而解決了許多常見(jiàn)的編程問(wèn)題。雖然閉包在JavaScript中有著廣泛的應(yīng)用,但也需要小心使用,以避免潛在的內(nèi)存泄漏問(wèn)題。確保在不再需要閉包時(shí),及時(shí)釋放對(duì)其的引用,以幫助垃圾回收器正常運(yùn)作。希望本文對(duì)你理解JavaScript中的閉包和如何應(yīng)用它們有所幫助。

本文轉(zhuǎn)載自微信公眾號(hào)「黑土豆的前端博客」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系公眾號(hào)。

責(zé)任編輯:武曉燕 來(lái)源: 黑土豆的前端博客
相關(guān)推薦

2023-09-06 16:55:33

JavaScript閉包

2023-02-19 15:22:22

React技巧

2022-01-12 15:50:24

JavaScript開(kāi)發(fā)循環(huán)

2022-05-04 12:44:57

Python編程語(yǔ)言

2022-12-22 14:44:06

JavaScript技巧

2022-12-25 16:03:31

JavaScript技巧

2024-11-14 09:00:00

Python編程元編程

2022-06-27 23:31:01

JavaScript框架開(kāi)發(fā)

2023-07-18 07:56:31

工具reduce業(yè)務(wù)

2023-06-28 00:02:40

2012-11-29 10:09:23

Javascript閉包

2020-06-21 13:57:21

JavaScript開(kāi)發(fā)代碼

2022-09-20 15:33:35

JavaScriptCSS編程

2009-03-13 09:39:34

JavaScript函數(shù)調(diào)用規(guī)則

2021-08-23 10:37:14

Javascript 機(jī)器學(xué)習(xí)阿里云

2022-05-30 09:44:11

TypeScriptJavaScript技巧

2023-09-07 16:28:46

JavaScrip

2023-11-02 08:53:26

閉包Python

2011-05-12 18:26:08

Javascript作用域

2023-11-13 10:00:09

數(shù)據(jù)中心服務(wù)器
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)