js腳本加載失敗問(wèn)題解決辦法
當(dāng)我們?cè)谶\(yùn)行某一個(gè)項(xiàng)目時(shí),它的某些JavaScript腳本可能加載不出來(lái),報(bào)錯(cuò)。 像現(xiàn)在單頁(yè)界面應(yīng)用基本上都是通過(guò)js來(lái)構(gòu)建,一旦加載不出,這個(gè)項(xiàng)目就不能運(yùn)行了,那怎么辦?我們?nèi)绾谓鉀Q這個(gè)問(wèn)題?
這里采用的方案是,重新加載JavaScript腳本。那么什么時(shí)候去重新加載js,如何重新加載js?在js腳本加載不出來(lái)時(shí),就需要去重試加載。那我們?cè)趺粗纉s有沒(méi)有加載失敗,我們可以在<script src="https://cdn.topskys.org/v0/loadjs.js"></script>標(biāo)簽上添加一個(gè)監(jiān)聽(tīng)加載失敗的事件 οnerrοr="console.log('error')",當(dāng)出現(xiàn)錯(cuò)誤會(huì)觸發(fā)onerror事件。
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> </head> <body> <script onerror="console.log('error')" src="https://cdn.topskys.org/v0/loadjs.js"></script> </body> </html>
可以看到,當(dāng)加載js出現(xiàn)錯(cuò)誤時(shí),onerror監(jiān)聽(tīng)到了錯(cuò)誤。但是在工程化的環(huán)境,這些<script></script>標(biāo)簽元素都是自動(dòng)生成的,給這些標(biāo)簽添加上onerror事件,這種方式又特別麻煩。我們可以給整個(gè)window添加error監(jiān)聽(tīng)事件:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> </head> <body> <script src="https://cdn.topskys.org/v0/loadjs.js"></script> <script> window.addEventListener("error",e=>{ console.log("error loading") }); </script> </body> </html>
但上一個(gè)js腳本加載出現(xiàn)錯(cuò)誤,window監(jiān)聽(tīng)error事件還有沒(méi)有注冊(cè),所以說(shuō)我們需要把它寫在最前邊,放到<head></head>里,一開(kāi)始就去監(jiān)聽(tīng)。當(dāng)然,寫到最前邊它也不會(huì)觸發(fā),因?yàn)樗粫?huì)事件冒泡,只能在捕獲階段就拿到該事件,在監(jiān)聽(tīng)事件函數(shù)后加“, true”:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> <script> window.addEventListener("error", e => { console.log("error loading") },true); // 捕獲階段拿到事件 </script> </head> <body> <script src="https://cdn.topskys.org/v0/loadjs.js"></script> </body> </html>
觸發(fā)error的事件類型也很多,如圖片加載錯(cuò)誤、在js中throw 1報(bào)錯(cuò)也會(huì)觸發(fā)error。我么怎么樣縮小觸發(fā)error的范圍,只監(jiān)聽(tīng)到那些<script src=""></script> 加載js不出來(lái)的情況?我們發(fā)現(xiàn)error事件中,參數(shù)e輸入的target是script元素:
但是呢,throw 在error事件里報(bào)的是ErrorEvent錯(cuò)誤,里面包含各種屬性,故我們可以準(zhǔn)確地找到某一種錯(cuò)誤。
<script> throw 1; </script>
我們需要在window監(jiān)聽(tīng)error事件里面作一個(gè)判斷:
<head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> <script> window.addEventListener("error", e => { const tag=e.target if(tag.tagName==='SCRIPT' && !(e instanceof ErrorEvent)) { console.log("JS 加載錯(cuò)誤") } },true); </script> </head>
到這里頁(yè)面解決了,什么時(shí)候去重新加載js的問(wèn)題,還剩下如何重試加載js。
首先,我們需要引入多個(gè)新地址的script標(biāo)簽加載js,當(dāng)某一個(gè)js加載失敗時(shí),就會(huì)被觸發(fā)window監(jiān)聽(tīng)事件error,執(zhí)行重試更換域名生成新的script標(biāo)簽加載js,需要靠document.write()才能阻塞后面的js加載,不阻塞后續(xù)js加載,會(huì)造成js加載順序混亂。
注意,在使用document.write()寫入script標(biāo)簽時(shí),需要對(duì)標(biāo)簽結(jié)束符進(jìn)行轉(zhuǎn)譯,否則會(huì)被認(rèn)為是上個(gè)標(biāo)簽的結(jié)束符。
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> <script> const domains = ["https://cdn.topskys.org/v0", "https://cdn.topskys1.org/v1", "https://cdn.topskys2.org/v2"]; // 重試域名數(shù)組 const maxRetry=3; // 最多重試次數(shù) const retryInfo={}; // 記錄重試次數(shù)及下一個(gè)重試的域名數(shù)組下標(biāo){times:1,nextIndex:2} window.addEventListener("error", e => { const tag = e.target if (tag.tagName === 'SCRIPT' && !(e instanceof ErrorEvent)) { // console.log("JS 加載錯(cuò)誤") const url=new URL(tag.src) // 拿到script標(biāo)簽的src域名 if(!retryInfo[url.pathname]){ retryInfo[url.pathname]={ times:0, nextIndex:0, } } const info=retryInfo[url.pathname]; // 取出文件路徑 ./loadjs.js ./js.js ./y.js const script=document.createElement("script") url.host=domains[info.nextIndex]; // 更換域名 document.write(`<script src="${url.toString()}">\<\/script>`); // 新加載script元素,需要阻塞后面的script加載js,否則加載js順序會(huì)亂。注意轉(zhuǎn)譯</script>,否則會(huì)被認(rèn)為上面的script結(jié)束標(biāo)簽。 script.src=url.toString(); document.body.insertBefore(script,tag); // 將新的script標(biāo)簽插入加載錯(cuò)誤標(biāo)簽前 // 修改重試信息 info.times++; info.nextIndex=(info.nextIndex+1)%domains.length; } }, true); </script> </head> <body> <script src="https://cdn.topskys.org/v0/js.js"></script> <script src="https://cdn.xx1.cn/y.js"></script> <script src="https://cdn.xx2.com/loadjs.js"></script> </body> </html>
那么,如何解決js加載失敗的問(wèn)題就解決了。但如果script標(biāo)簽上有defer、async及工程化項(xiàng)目中呢,有如何該解決?
總結(jié)
到此這篇關(guān)于js腳本加載失敗問(wèn)題解決辦法的文章就介紹到這了,更多相關(guān)js腳本加載失敗內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
php register_shutdown_function函數(shù)詳解
register_shutdown_function() 函數(shù)可實(shí)現(xiàn)當(dāng)程序執(zhí)行完成后執(zhí)行的函數(shù),其功能為可實(shí)現(xiàn)程序執(zhí)行完成的后續(xù)操作,需要的朋友可以參考下2017-07-07js實(shí)現(xiàn)的捐贈(zèng)管理完整實(shí)例
這篇文章主要介紹了js實(shí)現(xiàn)的捐贈(zèng)管理完整實(shí)例,包括了html頁(yè)面、js腳本及css樣式的完整實(shí)現(xiàn)代碼,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-01-01js實(shí)現(xiàn)拉伸拖動(dòng)iframe的具體代碼
這篇文章介紹了js實(shí)現(xiàn)拉伸拖動(dòng)iframe的具體代碼,有需要的朋友可以參考一下2013-08-08js中的關(guān)聯(lián)數(shù)組與普通數(shù)組詳解
下面小編就為大家?guī)?lái)一篇js中的關(guān)聯(lián)數(shù)組與普通數(shù)組詳解。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-07-07一文詳解JavaScript的事件監(jiān)聽(tīng)(最新整理)
Web頁(yè)面需要經(jīng)常和用戶之間進(jìn)行交互,而交互的過(guò)程中我們可能想要捕捉這個(gè)交互的過(guò)程,比如用戶點(diǎn)擊了某個(gè)按鈕、用戶在輸入框里面輸入了某個(gè)文本、用戶鼠標(biāo)經(jīng)過(guò)了某個(gè)位置,下面介紹下JavaScript的事件監(jiān)聽(tīng),感興趣的朋友一起看看吧2024-01-01js自定義trim函數(shù)實(shí)現(xiàn)刪除兩端空格功能
這篇文章主要介紹了js自定義trim函數(shù)實(shí)現(xiàn)刪除兩端空格功能,結(jié)合實(shí)例形式分析了javascript基于正則替換實(shí)現(xiàn)類似trim函數(shù)刪除字符串兩端空格的相關(guān)操作技巧,并附帶jQuery類似功能函數(shù)使用方法,需要的朋友可以參考下2018-02-02使用JS實(shí)現(xiàn)在空白頁(yè)上展示出一個(gè)有趣的時(shí)鐘
在我們?nèi)粘5木W(wǎng)頁(yè)瀏覽中,空白的頁(yè)面往往會(huì)被視為一種無(wú)趣的事物,一片等待填充的空間,今天我們來(lái)學(xué)習(xí)一下如何使用JS在空白的網(wǎng)頁(yè)上展示出一個(gè)有趣的時(shí)鐘吧,感興趣的小伙伴跟著小編一起來(lái)看看吧2024-04-04