JavaScript異步隊列進行try?catch時的問題解決
一、前言
我們在寫js的時候,經(jīng)常的會遇到需要異步去請求接口,或者通過setTimeout或Promise去做什么事, 然后讓同步進程繼續(xù)向下走, 當?shù)侥硞€時間節(jié)點的時候或者數(shù)據(jù)請求成功的時候在通過eventloop
的方式回調(diào)執(zhí)行。這本身是js的特點和優(yōu)勢。
但是,異步隊列執(zhí)行也存在錯誤的情況,這時,對于怎么進行錯誤處理,就成了我們的重點。
想一下項目中用到的方式,或者jquery的ajax方式,一般都會有catch、fail之類的回調(diào)方法供我們對錯誤結(jié)果進行處理。 那么現(xiàn)在討論的話題是能不能使用try catch
進行處理。
為什么寫這篇文章? 是因為我在寫JavaScript 的setTimeout與事件循環(huán)機制event-loop的時候,舉例express的異步錯誤獲取的時候,想到了這個點,我覺得有必要單獨拿出來,寫一篇斷篇幅的,又能夠清晰明了表達的一篇文章。于是這篇文章便生成了。
好了, 正文開始。
二、主要講的異步隊列方法
2.1 setTimeout
這里的setTimeout指的是一類,包括 setTimeout
, setInterval
這類所謂宏任務。 他們可以用try catch來捕獲錯誤么?
2.1.1 問題表現(xiàn)
try{ setTimeout(() => { let a = c; }, 100) } catch(e) { console.log('能獲取到錯誤么??', e); }
結(jié)果是不能獲取到,程序直接報錯了, 那么出現(xiàn)的后果可能就是整個頁面掛了,或者在node中,整個服務掛了。 我們的初心是想讓程序更加健壯,但卻做了無用功。
那么我們在想,既然在setTimeout 外邊無法獲取,那么能不能在setTimeout里面先用try catch獲取一下,然后捕獲到錯誤后再傳出去呢? 想到就干,繼續(xù)分析:
try{ setTimeout(() => { try { let a = c; } catch(e) { throw new Error('some variable is not defined'); } }, 100) } catch(e) { console.log('能獲取到錯誤么??', e); }
很抱歉,想法很好,但是也不行。外邊也catch不到
。
2.1.2 問題原因
好了,我們把這個疑問分析一下吧。其實,這里的根本原因還是剛開始提到的事件循環(huán)
。 事件循環(huán)不是空空的一句表述、一個概念,而是在代碼中實實在在存在的。
具體事件循環(huán)的相關知識,可以看下我很早前寫的JavaScript 的setTimeout與事件循環(huán)機制event-loop 文章。
回到這個例子中, 最外層的try catch是在一個task
中,我們定義它為我們js文件的同步主任務,從上到下執(zhí)行到這里了, 然后,會把里面的setTimeout推到一個任務隊列
中, 這個隊列是存儲在內(nèi)存中的,由V8來管理。然后主task就繼續(xù)向下執(zhí)行, 一直到結(jié)束。
當該setTimeout時間到了,且沒有其它的task執(zhí)行了, 那么,就將這個setTimeout的代碼推入執(zhí)行棧
開始執(zhí)行。 當執(zhí)行到錯誤代碼的時候,也就是這個 let a = c
, 因為c未定義,所以就會報錯。
但問題的本質(zhì)是,這個錯誤跟最外層的try catch并不在一個執(zhí)行棧中,當里面執(zhí)行的時候,外邊的這個task早已執(zhí)行完, 他們的context(上下文)已經(jīng)完全不同了。
所以,頁面會直接報錯,甚至程序崩潰。
2.2 Promise
我們知道,Promise
也是一個異步的處理過程,它對應事件循環(huán)中的微任務
。 那么這里其實與上面的setTimeout存在同樣的問題。
舉個例子:
try { new Promise((resolve, reject) => { reject('promise error'); }) } catch(e) { console.log('異步錯誤,能catch到么??', e); }
相信大家能夠推導出結(jié)果了: 也catch不到
原因其實與上面的setTimeout是一樣的,執(zhí)行棧上下文已經(jīng)不同了。
那么針對Promise,ECMA官方已經(jīng)給我們提供了一個方法,那就是 catch
, 通過catch我們獲取到錯誤,可以阻止程序崩潰。 但是喜歡發(fā)散思維的你可能會想到, 那我用catch接到了,是不是就可以讓外層的catch獲取到了呢? 想到就試一下
try { new Promise((resolve, reject) => { reject('promise error'); }).catch(e => { throw new Error(e); }) } catch(e) { console.log('異步錯誤,能catch到么??', e); }
結(jié)果就是 不行
。相信大家通過我詳細的例子和思維脈絡,對這塊已經(jīng)真正掌握了吧?
2.3 callback
那么通過上面的,大家可能會有一種想法,只要是callback,就是catch不住的。 其實這種想法是錯誤的,我通過一個例子來證明。
function Fn(cb) { console.log('callback執(zhí)行了'); cb(); } try { const cb = () => { throw new Error('callback執(zhí)行錯誤'); } Fn(cb); } catch(e) { console.log('能夠catch住么???') }
其實這里就是個煙霧彈, 考驗大家對這個事件循環(huán)相關機制是不是明白了。
2.4 Async await
現(xiàn)在的項目中,大家越來越愿意使用Async await
這對 es7標準里的api了。 因為它們這對組合是在是太好用了。 那么通過異步等待的方式,用try catch能夠行么?
那么咱們使用一個例子驗證一下:
const asyncFn = () => { return new Promise((resolve, reject) => { setTimeout(() => { reject('asyncFn執(zhí)行時出現(xiàn)錯誤了') }, 100); }) } const executedFn = async () => { try{ await asyncFn(); }catch(e) { console.log('攔截到錯誤..', e); } }
如果執(zhí)行一下,就發(fā)現(xiàn): catch到了!
asyncFn
里面是有 Promise的,為什么外邊就能catch到了呢? 是不是跟上面講的矛盾了呢? 其實并沒有。 看我分析一下:
async-await 是使用生成器、promise 和協(xié)程實現(xiàn)的,wait操作符還存儲返回事件循環(huán)之前的執(zhí)行上下文,以便允許promise操作繼續(xù)進行。當內(nèi)部通知解決等待的承諾時,它會在繼續(xù)之前恢復執(zhí)行上下文。
所以說,能夠回到最外層的上下文, 那就可以用try catch 啦。
到此這篇關于JavaScript異步隊列進行try catch時的問題解決的文章就介紹到這了,更多相關JS try catch內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
JS中JSON對象和String之間的互轉(zhuǎn)及處理技巧
JSON:JavaScript 對象表示法(JavaScript Object Notation),其實JSON就是一個JavaScript的對象(Object)而已。接下來通過本文給大家介紹JS中JSON對象和String之間的互轉(zhuǎn)及處理技巧,需要的朋友一起學習吧2016-04-04JavaScript在IE和Firefox(火狐)的不兼容問題解決方法小結(jié)
今天測試代碼時,發(fā)現(xiàn)不少IE可以運行的ajax,但在FF中報錯。IE和Firefox(火狐)在JavaScript方面的不兼容及統(tǒng)一方法總結(jié)如下,需要的朋友可以看下,對于以后的代碼書寫一定要考慮到多瀏覽器的兼容性。2010-04-04使用JavaScript修改瀏覽器URL地址欄的實現(xiàn)代碼
這篇文章主要介紹了如何使用JavaScript修改瀏覽器URL地址欄,需要的朋友可以參考下2013-10-10