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

Javascript的異步函數(shù)和Promise對象你了解嗎

 更新時間:2022年03月11日 15:21:02   作者:小綿楊Yancy  
這篇文章主要為大家詳細介紹了Javascript異步函數(shù)和Promise對象,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助

1、JS中的異步

1.1 同步

一般情況下,js的代碼都是自上而下順序運行的。例如:

let res = '';
res = '獲取到的結(jié)果!';
console.log(res);

結(jié)果:

在這里插入圖片描述

很容易理解,我給res賦了新值,然后輸出res。這就是js的同步執(zhí)行,這里的同步,并不是一起執(zhí)行的意思,而是在一個線程里順序執(zhí)行的意思。因為JavaScript是單線程,所以所有程序都應該在一個線程里運行。

1.2 異步

但是有的時候,我們獲取res的值是需要一點時間的,例如使用ajax向服務器發(fā)起請求,服務器響應以后返回結(jié)果,我們再將結(jié)果賦值給res。

這里使用setTimeout函數(shù)模擬數(shù)據(jù)請求,setTimeout也是一個異步函數(shù)。

let res = '';
setTimeout(()=>{
    res = '獲取到的結(jié)果!';
    console.log('獲取到結(jié)果了!',res);
},3000);
console.log('res',res);

在這里插入圖片描述

可以看到,沒有立刻獲取到結(jié)果,而是3s后才獲取到結(jié)果。

為什么會這樣呢?
由于獲取res它是一個異步操作,所以它會被分為兩部分來執(zhí)行,先調(diào)用setTimeout方法,然后把要執(zhí)行的函數(shù)放到一個隊列中。代碼繼續(xù)往下執(zhí)行,當把所有的代碼都執(zhí)行完后,放到隊列中的函數(shù)才會被執(zhí)行。因為js的單線程機制,不允許它花費時間去等待異步函數(shù)的結(jié)果。

1.3 回調(diào)函數(shù)解決異步問題

既然異步函數(shù)的結(jié)果會再最后獲取,那么我們就可以給異步函數(shù)中加一個回調(diào)函數(shù),來處理獲取到的數(shù)據(jù)。

let res = '';
setTimeout(()=>{
    res = '獲取到的結(jié)果!';
    callback();
},3000);
function callback(){
    console.log('獲取到結(jié)果了!',res);
    console.log('處理結(jié)果!');
}
console.log('res', res);

在這里插入圖片描述

現(xiàn)在console.log('res', res);仍然沒有獲取到res,但是我們已經(jīng)不需要它了,我們獲取res的就是為了處理它,而使用callback函數(shù)就可以達到目的了!

1.4 回調(diào)地獄

1.3了解了可以通過調(diào)用函數(shù)解決無法獲取結(jié)果的問題,但是它仍然是存在缺點的,如果只獲取一次結(jié)果,那還好。但是如果我們在獲取結(jié)果之后,還需要利用獲取的結(jié)果再進行異步操作,那么又需要嵌套一層,又需要一次異步操作,再嵌套一層……

setTimeout(function(){
    console.log("first");
    setTimeout(function(){
        console.log("second");
        setTimeout(function(){
            console.log("third");
            setTimeout(function(){
                console.log("fourth");
            },2000);
        },2000);
    },2000);
},2000);

雖然上述代碼會按照我們預期的方向運行,但是多層的嵌套讓代碼閱讀和維護起來十分的困難。

2、Promise對象

2.1 Promise的基本使用

為了解決異步中的回調(diào)地獄問題,ES6引入了Promise對象,使得我們可以十分優(yōu)雅地進行異步操作。
從語法上來說,Promise是一個對象,從它可以獲取異步操作的消息。

let timeout = function(time){
    return new Promise(function(resolve,reject){
        setTimeout(function(){
            resolve();
        },time);
    });
}
console.log("開始運行!");
timeout(2000).then(function(){
    console.log("first");
    return timeout(2000);
}).then(function(){
    console.log("second");
    return timeout(2000);
}).then(function(){
    console.log("third");
    return timeout(2000);
}).then(function(){
    console.log("fourth");
    return timeout(2000);
});

在這里插入圖片描述

這樣就解決了上述的回調(diào)地獄的問題。并且then也很容易理解,就是上一個異步函數(shù)執(zhí)行完成后,接著要進行的操作。

同時Promise對象也可以通過resolve和reject傳遞參數(shù):

let res = null;
let timeout = function(time){
    return new Promise(function(resolve,reject){
        setTimeout(function(){
            res = false;
            if(res){
                resolve('res為true');
            }else{
                reject('res為false');
            }
        },time);
    });
}
timeout().then((res)=>{
    console.log(res);
}).catch((error)=>{
    console.log(error);
})

這樣就可以根據(jù)res的值來確定結(jié)果了。resolve()對應then的結(jié)果,而reject()對應catch的結(jié)果。這在axios的請求操作中是十分常見的。

2.2 async 和 await

mdn描述如下:

async函數(shù)是使用async關鍵字聲明的函數(shù)。 async函數(shù)是AsyncFunction構(gòu)造函數(shù)的實例, 并且其中允許使用await關鍵字。async和await關鍵字讓我們可以用一種更簡潔的方式寫出基于Promise的異步行為,而無需刻意地鏈式調(diào)用promise。

async和await的關系:

async函數(shù)可能包含0個或者多個await表達式。await表達式會暫停整個async函數(shù)的執(zhí)行進程并出讓其控制權(quán),只有當其等待的基于promise的異步操作被兌現(xiàn)或被拒絕之后才會恢復進程。promise的解決值會被當作該await表達式的返回值。使用async / await關鍵字就可以在異步代碼中使用普通的try / catch代碼塊。

function getProcessedData(url) {
  return downloadData(url) // 返回一個 promise 對象
    .catch(e => {
      return downloadFallbackData(url)  // 返回一個 promise 對象
    })
    .then(v => {
      return processDataInWorker(v); // 返回一個 promise 對象
    });
}

使用async和await重寫上述函數(shù)。

async function getProcessedData(url) {
  let v;
  try {
    v = await downloadData(url);
  } catch (e) {
    v = await downloadFallbackData(url);
  }
  return processDataInWorker(v);
}

總結(jié)

本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關注腳本之家的更多內(nèi)容! 

相關文章

  • JavaScript資源預加載組件和滑屏組件的使用推薦

    JavaScript資源預加載組件和滑屏組件的使用推薦

    這篇文章主要介紹了JavaScript資源預加載組件和滑屏組件的使用推薦,分別為preload和slide的用法講解,使用起來非常簡單,需要的朋友可以參考下
    2016-03-03
  • JavaScript+CSS實現(xiàn)的可折疊二級菜單實例

    JavaScript+CSS實現(xiàn)的可折疊二級菜單實例

    這篇文章主要介紹了JavaScript+CSS實現(xiàn)的可折疊二級菜單,以完整實例形式分析了JavaScript基于頁面元素節(jié)點及樣式的動態(tài)操作實現(xiàn)折疊菜單的相關技巧,需要的朋友可以參考下
    2016-02-02
  • js實現(xiàn)常見的工具條效果

    js實現(xiàn)常見的工具條效果

    本文主要介紹了js實現(xiàn)常見的工具條效果的實例。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-03-03
  • js制作簡易計算器

    js制作簡易計算器

    這篇文章主要為大家詳細介紹了js制作簡易計算器,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • JavaScript中的對象的extensible屬性介紹

    JavaScript中的對象的extensible屬性介紹

    這篇文章主要介紹了JavaScript中的對象的extensible屬性介紹,JavaScript中,對象的extensible屬性用于表示是否允許在對象中動態(tài)添加新的property,需要的朋友可以參考下
    2014-12-12
  • 善用事件代理,警惕閉包的性能陷阱。

    善用事件代理,警惕閉包的性能陷阱。

    關于JS性能優(yōu)化中的冰山一角:事件代理、警惕閉包。其實本文有一個文章已經(jīng)說到,閉包如何產(chǎn)生,閉包的作用;
    2011-01-01
  • JavaScript 事件參考手冊

    JavaScript 事件參考手冊

    對于js的一些常見事件的總結(jié)
    2008-12-12
  • 使用Web?Component實現(xiàn)防篡改水印

    使用Web?Component實現(xiàn)防篡改水印

    Web?Component內(nèi)部有鉤子天然支持被篡改時被觸發(fā),用來防篡改非常方便,所以本文就將使用Web?Component實現(xiàn)防篡改水印,感興趣的小伙伴可以了解下
    2023-12-12
  • 老生常談jacascript DOM節(jié)點獲取

    老生常談jacascript DOM節(jié)點獲取

    下面小編就為大家?guī)硪黄仙U刯acascript DOM節(jié)點獲取。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-04-04
  • 滾動條代碼生成器

    滾動條代碼生成器

    滾動條代碼生成器...
    2007-02-02

最新評論