溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么在Javascript中實現異步等待

發布時間:2021-05-24 15:47:08 來源:億速云 閱讀:719 作者:Leah 欄目:開發技術

本篇文章給大家分享的是有關怎么在Javascript中實現異步等待,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

async/await 是javascript中的一種模式,可使您的代碼以同步方式執行,但又不影響javascript的異步行為。

定義異步功能

要定義一個異步函數,您所要做的只是在函數定義之前添加一個async關鍵字。

// async function always returns a promise
async function greet() {
  return "hello";
}

輕松自在!?。在函數名稱前使用async關鍵字

使該函數返回一個承諾。

函數返回時解析。

拋出錯誤時最終拒絕。

這意味著您每次要創建一個Promise時都不需要聲明返回Promise.new()。

為了證明異步函數返回了一個Promise,我們可以快速附加一個then塊以打印其值。

async function greet() {
  return "Hello from an async function"
}
greet().then(message => console.log(message));
//Hello from an async function

使用等待和執行異步功能

不冷靜,我們可以做的then(),并catch()在一個async功能?但這不是異步async函數的真正功能,函數的真正潛力在于await語句。

await 使函數以同步方式執行,同時將控件保持在該行中,直到等待方法完成其執行。

async function greet() {
  return "Hello from an async function"
}

async function execute() {
  const message = await greet();
  console.log(message)
}

這是我們需要記住的一些經驗法則。

?等待只能在異步函數內使用

async如果我們在函數內部使用await,則必須聲明一個函數,反之則不然。

讓我這樣說。如果await在方法內部使用語句,則該方法必須是async方法,否則編譯器會大吼大叫。

async function greet() {
  return "Hello from an async function";
}

function execute() {//this function must be async
  const message = await greet();
  console.log(message)
}
/* 
SyntaxError: await is only valid in async function
*/

但是聲明一個函數async并不一定意味著我們將始終await在其內部使用它。這greet()是一個async方法,但是await里面沒有任何語句。

wait當調用函數,返回promise或為異步函數時,await才有意義

//not an async function
function greet() {
 return "Hello from an async function";
}

async function execute() {
  const message = await greet();
  console.log(message); //Hello from an async function
}

盡管代碼的工作原理與上一代碼完全相同,但是await對synchronous函數進行操作沒有任何意義。我想知道您對此有何想法?

使用await的一個重要方面是它阻塞了下一行代碼的執行,直到執行await塊為止。

const asyncGreet = () => new Promise(resolve => setTimeout(resolve, 2000));

(async function execute() {
  console.log("before executing");
  await asyncGreet(); //blocks execution here
  // ? executed once await is finished
  console.log("I will be executed after 2000ms");
})();

現在您必須懷疑是否正在等待使代碼同步,為什么我們要使用它呢?NodeJ或瀏覽器Javascript是單線程環境,一次執行一項任務,由于它們的異步行為而被廣泛使用,而我們正在失去這些行為。那有什么意義呢?

是的,您是對的,但是如果您在大多數情況下都觀察到,我們需要執行與他人有關的任務。

async function subscribeToNewsLetter() {
  const user  = await findUser(id);
  //?methods need user email to execute
  await subscribe(user.email)
  await sendNotification(user.email)
}

沒錯 但是互不相關的代碼呢?好吧,還有一個替代方法,即(Promise.all)。

const asyncGreet = (name) =>  new Promise((resolve) => setTimeout(resolve(`Hello ${name}`), 2000));

const names = ['john', 'jane', 'david'];

(async function() {
  const greetingPromises = names.map(name => asyncGreet(name));
  console.log(await Promise.all(greetingPromises));
})();

我知道上面的代碼是一個人為的示例,在這里重要的是我們正在利用的力量Promise.all來執行所有的諾言

處理錯誤Async/Await。

使用async / await處理錯誤非常容易,我們可以使用我們的老朋友try / catch塊來實現這一點。

async function subscribeToNewsLetter() {
  try {
    const user  = await findUser(id);
    await subscribe(user.email)
    await sendNotification(user.email)
  } catch(err) {
    //handle error
  }
}

還有另一個版本,我們可以將catch處理程序直接附加到await塊。我個人不使用它,但是如果您愿意,可以嘗試一下。

  await asyncGreet().catch(err => console.log(err);

2倍的可讀性,易于調試

以下代碼使用Promise通過id查找用戶,分配配置文件信息,然后查找用戶的訂閱。

function getUser(id, profile) {
  return new Promise((resolve, reject) => {
    User
      .find(id)
      .then((user) => {
        if(_.isEmpty(user)) return {};
        user.profile = profile;
        return user;
      })
      .then((user) => Subscription.find(user.id))
      .then(subscription => {
        if(_.isEmpty(subscription)) {
          user.subscription = null;
        } else {
          user.subscription = subscription;
        }
        return resolve(user)
      })
      .catch(err => reject(err))
  })
}

上面的代碼工作完全正常,但我們肯定可以使其更具可讀性,簡潔,易于調試與async/ await。讓我們去吧。

async function getUser(id, profile) {
  try {
    const user = await User.find(id);
    if(_.isEmpty(user)) return {};
    user.profile = profile;
    const subscription = await Subscription.find(user.id);
    user.subscription = subscription
    return user;
  } catch(err) {
    console.log(err);
  }
}

回調和Async/Await是敵人

正如我們在前面的示例中已經看到的那樣,promise與async/一起使用非常好await。任何返回promise的函數都可以與await語句一起使用。

但是當涉及到回調時,情況恰恰相反,回調不能直接與async/一起使用await,必須將它們轉換為Promise。

讓我們考慮以下函數,該函數異步測試值是否為偶數(引發錯誤)。

function asyncEven(id, cb){
  setTimeout(() => {
    const even = id%2 === 0;
    if (even) return cb(null, "even");
    else return cb("not even");
  }, 2000);
}

我們知道在回調中不允許使用await,但是仍然嘗試一下。

(async function() {
  //?? Wrong way
  const even = await asyncEven(2);
  console.log("isEven ", even); //undefined
})();

您一定在想,我們沒有附加一個回調,這就是它打印的原因undefined。

讓我們附加一個回調,這是很奇怪的,但是讓我們有耐心。

(async function() {
  //this is also wrong ??
  const even = await asyncEven(2, (err, data) => { console.log("inside await on callback", err, data)});
  console.log("isEven ", even);
})();
/*
output:
even  undefined
inside await on callback even null
*/

似乎調用了回調,并且我們還從asyncEven函數中獲取了值。沒錯,但這仍然是錯誤的方法。

await對回調沒有影響。這類似于在同步功能上進行等待。

那為什么它返回undefined呢?這是個好問題。這是異步編程的默認性質。該setTimeout的功能是一個回調返回通過回調值2000毫秒之后,同時,控制開始執行的下一行代碼,并且它達到的功能,這就是為什么我們得到的最終未定義。

那么解決方案是什么?很簡單 將asyncEven功能變為承諾并await像冠軍一樣使用。

function asyncEven(id,) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      const even = id%2 === 0;
      if (even) return resolve("even");
      else return reject("not even");
    }, 2000);
  })
}

(async function() {
  // waits for the execution
  const even = await asyncEven(2);
  console.log("iseven ", even);
})();

ForEach不適合與 Async/Await

如果我們將ForEach循環與一起使用,則可能會有副作用async/await??紤]以下示例,console.log此處的語句不等待await

greet(name)。
async function greet(name) {
 return Promise.resolve(`Hello ${name}, how are you ?`);
}

(function() {
  console.log("before printing names");
  const names = ['john', 'jane', 'joe'];
  names.forEach(async (name) => {
   //does not wait here
    console.log(await greet(name));
  });
  console.log("after printing names");
})();
/*
before printing names
after printing names
Hello john, how are you ?
Hello jane, how are you ?
Hello joe, how are you ?
*/

不僅僅是語法糖

到目前為止,我們只知道這async/await使我們的代碼更具可讀性,調試友好性,并且有人說這是javascript promise的語法糖。實際上,它不只是語法糖。

// promise
async1()
.then(x => asyncTwo(x))
.then(y => asyncThree(y))
//other statement
console.log("hello")


//async await
x = await async1();
y = await asyncTwo(x);
await asyncThree(y);

await暫停當前函數的執行,而promise繼續執行當前函數,將值添加到中then()。這兩種執行程序的方式之間存在顯著差異。

讓我解釋一下,考慮諾言版本,如果asyncTwo()或asyncThree()在執行任務時拋出異步錯誤,它將包含async1()在堆棧跟蹤中嗎?

這里的promise不會暫停當前函數的執行,當asyncTwo解析或拒絕時,上下文不在promise語句之內。因此,理想情況下,它不能包含asyncOne在堆棧跟蹤中。但是由于使用了V8引擎,它在這里做了一些神奇的工作,通過asyncOne()提前引用以便包含asyncOne()在上下文中。但這不是免費的。捕獲堆棧跟蹤需要花費時間(即降低性能)。存儲這些堆棧跟蹤需要內存。

async/await在性能方面,這是拍子承諾的地方,因為當前功能的執行將暫停,直到等待功能完成為止,因此我們已經對該功能有了參考。

JavaScript的作用是什么

1、能夠嵌入動態文本于HTML頁面。2、對瀏覽器事件做出響應。3、讀寫HTML元素。4、在數據被提交到服務器之前驗證數據。5、檢測訪客的瀏覽器信息。6、控制cookies,包括創建和修改等。7、基于Node.js技術進行服務器端編程。

以上就是怎么在Javascript中實現異步等待,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女