【01-17】defer柑肴、async作用和區(qū)別分析

瀏覽器內(nèi)核可以分成兩部分:渲染引擎(Layout Engine 或者 Rendering Engine)和 JS 引擎。早期渲染引擎和 JS 引擎并沒有十分明確的區(qū)分旬薯,但隨著 JS 引擎越來越獨(dú)立晰骑,內(nèi)核也成了渲染引擎的代稱(下文我們將沿用這種叫法)。渲染引擎又包括了 HTML 解釋器硕舆、CSS 解釋器骤公、布局、網(wǎng)絡(luò)阶捆、存儲、圖形倍奢、音視頻、圖片解碼器等等零部件卒煞。

JS 引擎是獨(dú)立于渲染引擎存在的叼架。我們的 JS 代碼在文檔的何處插入衣撬,就在何處執(zhí)行扮饶。當(dāng) HTML 解析器遇到一個 script 標(biāo)簽時,它會暫停渲染過程贴届,將控制權(quán)交給 JS 引擎。JS 引擎對內(nèi)聯(lián)的 JS 代碼會直接執(zhí)行占键,對外部 JS 文件還要先獲取到腳本元潘、再進(jìn)行執(zhí)行。等 JS 引擎運(yùn)行完畢翩概,瀏覽器又會把控制權(quán)還給渲染引擎,繼續(xù) CSSOM 和 DOM 的構(gòu)建牍鞠。 因此與其說是 JS 把 CSS 和 HTML 阻塞了评姨,不如說是 JS 引擎搶走了渲染引擎的控制權(quán)。

渲染引擎碰到j(luò)s就交出大權(quán)是因為他不知道js的內(nèi)容會不會對接下來的渲染有沒有影響吐句。但是我們引入js的時候是知道有沒有影響的,可以根據(jù)具體情況用三種方式之一加載js攀芯。

三種加載方式

image.png

1.正常模式
<script src="script.js"></script>
沒有 defer 或 async,瀏覽器會立即加載并執(zhí)行指定的腳本侣诺,“立即”指的是在渲染該 script 標(biāo)簽之下的文檔元素之前择葡,也就是說不等待后續(xù)載入的文檔元素,讀到就加載并執(zhí)行阻星。

2.async模式
<script async src="script.js"></script>
有 async,script.js會被異步加載妥箕,即加載和渲染后續(xù)文檔元素的過程將和 script.js 的加載并行進(jìn)行(異步)。當(dāng) script.js加載完整立即執(zhí)行script.js坎吻。執(zhí)行script.js時宇葱,html解析暫停。
從加載完成立即執(zhí)行來看黍瞧,async模式 執(zhí)行順序與寫的順序無關(guān),不保證執(zhí)行順序您机。

3.defer 模式
<script defer src="index.js"></script>
有 defer年局,script.js會被異步加載,即加載和渲染后續(xù)文檔元素的過程將和 script.js 的加載并行進(jìn)行(異步)仲闽。這一點(diǎn)與async模式一致兴喂。
不同的是當(dāng) script.js加載完成并不會立即執(zhí)行,而是在所有元素解析完成之后衣迷,DOMContentLoaded 事件觸發(fā)之前完成酱酬。因此它會按照寫的順序執(zhí)行。

// html 
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>defer-async</title>

    <script type="text/javascript" async src='./async1.js'></script>
    <script type="text/javascript" async src='./async2.js'></script>
    <script type="text/javascript" src='./normal.js'></script>
</head>
<body>
    <div id="warp">warp</div>
</body>
</html>

然后 async1.js 文件巨大(到底有多大汗菜,我是把jquery的壓縮版拷進(jìn)來了)挑社,然后最后加上 console.log('async1');
文件async2.js 和 normal.js 中分別是 console.log('async2'); 和 console.log('normal');
打開網(wǎng)頁控制臺顯示如下: async2 先加載完成就先執(zhí)行了。


image.png
<script type="text/javascript" defer src='./defer1.js'></script>
<script type="text/javascript" defer src='./defer2.js'></script>
<script type="text/javascript" src='./normal.js'></script>

同理菌瘪,defer1.js 里放了jquery的壓縮版源碼。defer2.js里只放了一句日志; 刷新網(wǎng)頁看下日志:
defer1 糜工、defer2還是按照順序執(zhí)行的录淡。

把a(bǔ)sync、defer都加上嫉戚,

    <script type="text/javascript" async src='./async1.js'></script>
    <script type="text/javascript" async src='./async2.js'></script>

    <script type="text/javascript" defer src='./defer1.js'></script>
    <script type="text/javascript" defer src='./defer2.js'></script>
    <script type="text/javascript" src='./normal.js'></script>

日志如下:


image.png

這個順序應(yīng)該不是固定的彬檀,符合normal最早,defer1會在 defer2之前的規(guī)矩凤覆。 至于async 和 defer的前后則要看本身js的加載以及dom樹的構(gòu)建時機(jī)吧盯桦。

三種方式適合什么時候用

growingwiththeweb 推薦優(yōu)先級依次是 async defer normal。拥峦。

  • 當(dāng)你的js是個獨(dú)立的模塊且不依賴任何js,使用 async;
  • 如果你的js依賴其他js或者被其他js 依賴,使用 defer;
  • 如果你對js文件很小且被 async script 依賴玄柠,使用正常模式的script且放在async script 前面。

可能的坑

雖然理論上defer按加載順序執(zhí)行宫患,但也有同學(xué)反映事實(shí)上并不是這樣这弧。。比如這位同學(xué)的問題:


image.png

我認(rèn)為這是涉及到 event loop的 task和微任務(wù)了皇帮。
"在現(xiàn)實(shí)當(dāng)中蛋辈,延遲腳本并不一定會按照順序執(zhí)行,也不一定會在 DOMContentLoaded 事件觸發(fā)前執(zhí)行,因此最好只包含一個延遲腳本瓢娜。" 《JavaScript 高級程序設(shè)計(第三版)》如是說礼预,所以腳本之間有依賴,最好使用一個異步腳本吧托酸。比如上面同學(xué)那個問題 可以改成這樣<script src="1.js"></script>.

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末励堡,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子应结,更是在濱河造成了極大的恐慌,老刑警劉巖揩慕,帶你破解...
    沈念sama閱讀 218,284評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件扮休,死亡現(xiàn)場離奇詭異,居然都是意外死亡玷坠,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評論 3 395
  • 文/潘曉璐 我一進(jìn)店門樟凄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來兄渺,“玉大人,你說我怎么就攤上這事》欤” “怎么了?”我有些...
    開封第一講書人閱讀 164,614評論 0 354
  • 文/不壞的土叔 我叫張陵庐扫,是天一觀的道長。 經(jīng)常有香客問我铅辞,道長萨醒,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,671評論 1 293
  • 正文 為了忘掉前任囤踩,我火速辦了婚禮晓褪,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘涣仿。我一直安慰自己,他們只是感情好好港,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,699評論 6 392
  • 文/花漫 我一把揭開白布媚狰。 她就那樣靜靜地躺著,像睡著了一般崭孤。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上遗锣,一...
    開封第一講書人閱讀 51,562評論 1 305
  • 那天嗤形,我揣著相機(jī)與錄音,去河邊找鬼笔咽。 笑死霹期,一個胖子當(dāng)著我的面吹牛叶组,可吹牛的內(nèi)容都是我干的历造。 我是一名探鬼主播船庇,決...
    沈念sama閱讀 40,309評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼鸭轮,長吁一口氣:“原來是場噩夢啊……” “哼橄霉!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起酪劫,我...
    開封第一講書人閱讀 39,223評論 0 276
  • 序言:老撾萬榮一對情侶失蹤覆糟,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后滩字,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,668評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡漓藕,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,859評論 3 336
  • 正文 我和宋清朗相戀三年挟裂,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片栗竖。...
    茶點(diǎn)故事閱讀 39,981評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡渠啤,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出份名,到底是詐尸還是另有隱情,我是刑警寧澤僵腺,帶...
    沈念sama閱讀 35,705評論 5 347
  • 正文 年R本政府宣布壶栋,位于F島的核電站,受9級特大地震影響丧没,放射性物質(zhì)發(fā)生泄漏锡移。R本人自食惡果不足惜呕童,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,310評論 3 330
  • 文/蒙蒙 一夺饲、第九天 我趴在偏房一處隱蔽的房頂上張望施符。 院中可真熱鬧,春花似錦戳吝、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,904評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至隘马,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間酸员,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,023評論 1 270
  • 我被黑心中介騙來泰國打工醇王, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留崭添,地道東北人。 一個月前我還...
    沈念sama閱讀 48,146評論 3 370
  • 正文 我出身青樓棘伴,卻偏偏與公主長得像屁置,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子蓝角,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,933評論 2 355

推薦閱讀更多精彩內(nèi)容