我有一些<script>元素,其中一些代码依赖于其他<script>元素中的代码。我看到defer属性在这里可以派上用场,因为它允许延迟代码块的执行。

为了测试它,我在Chrome上执行了这个:http://jsfiddle.net/xXZMN/。

<script defer="defer">alert(2);</script>
<script>alert(1)</script>
<script defer="defer">alert(3);</script>

然而,它提醒2 - 1 - 3。为什么它不提醒1 - 2 - 3?


当前回答

<脚本延迟> - 只要浏览器使用defer与脚本标记进行交互

它开始获取脚本文件,同时还并排解析HTML。 在这种情况下,脚本只在HTML解析完成后执行。

<脚本async> - 当浏览器与async交互脚本标记时

它在并排解析HTML时开始获取脚本文件。 但是在完全获取脚本时停止HTML解析,然后开始执行脚本,之后继续进行HTML解析。

<脚本> - - - 只要浏览器与脚本标记交互

它停止HTML的解析,获取脚本文件, 在本例中,执行脚本,然后继续解析HTML。

其他回答

看看谷歌开发人员Jake Archibald在2013年写的这篇深入研究脚本加载的文章。

引用该条有关部分:

Defer <script src="//other-domain.com/1.js" defer></script> <script src="2.js" defer></script> Spec says: Download together, execute in order just before DOMContentLoaded. Ignore “defer” on scripts without “src”. IE < 10 says: I might execute 2.js halfway through the execution of 1.js. Isn’t that fun?? The browsers in red say: I have no idea what this “defer” thing is, I’m going to load the scripts as if it weren’t there. Other browsers say: Ok, but I might not ignore “defer” on scripts without “src”.

(根据这条评论,我将添加早期版本的Firefox在延迟脚本完成运行之前触发DOMContentLoaded。)

现代浏览器似乎正确地支持异步,但您需要接受脚本无序运行,并且可能在DOMContentLoaded之前运行。

因为defer属性只适用于带有src的脚本标签。找到了一种方法来模拟内联脚本的延迟。使用DOMContentLoaded事件。

<script defer src="external-script.js"></script>
<script>
document.addEventListener("DOMContentLoaded", function(event) {
    // Your inline scripts which uses methods from external-scripts.
});
</script>

这是因为,DOMContentLoaded事件在延迟带属性脚本完全加载后触发。

延迟只能在<script>标记中用于外部脚本包含。因此,建议在<head>-section中的<script>-标签中使用。

还应该注意的是,在IE<=9中,在某些情况下使用脚本延迟可能会出现问题。更多相关信息:https://github.com/h5bp/lazyweb-requests/issues/42

defer属性仅用于外部脚本(只有在src属性存在时才应该使用)。