是的,script放在head里默认会阻塞页面渲染;浏览器自上而下解析HTML时,遇无async/defer的script即暂停DOM构建,导致白屏、首屏延迟及DOM操作失败(如document.getElementById返回null)。

script放在head里会阻塞页面渲染
浏览器的HTML解析器是“急性子”,它从上到下逐行解析,一旦遇到一个没有 async 或 defer 属性的 标签,就会立刻停下手中的活儿——也就是暂停DOM树的构建——转而去下载、解析并执行这个脚本。想象一下,如果这个脚本恰好放在 里,那意味着什么?整个 都还没开始处理,用户自然只能面对一片空白,这就是我们常说的“白屏”现象。
由此引发的典型错误,前端开发者应该都不陌生:document.getElementById(“app”) 返回 null,因为你要操作的那个DOM节点根本还没生成;或者页面首屏内容加载异常缓慢,在弱网环境下,这种延迟会被进一步放大。
- 内联脚本(无
src):即便放在,执行时机也很早,但风险极高,必须确保脚本逻辑不依赖于任何尚未存在的DOM元素。 - 外部脚本(带
src):默认是同步加载,它不仅会阻塞后续的DOM解析,甚至可能阻塞CSS、图片等其他资源的加载。 - 关于
defer与async:defer仅对外部脚本有效,并且严格保证多个脚本按照它们在HTML中间出现的顺序依次执行;而async则“各自为政”,不保证顺序,更适合那些完全独立、不依赖其他脚本的代码,比如统计脚本。
script放在body底部能安全操作DOM
将 标签放在 闭合标签之前,是一个经典且有效的策略。这相当于告诉浏览器:“先把页面的骨架和内容都搭建好,最后再来处理这些脚本。”此时,document.body 以及页面上的常规元素都已准备就绪。这时再去调用 getElementById、绑定事件监听器,或者初始化像Vue这样的框架(new Vue({ el: “#app” })),基本可以确保万无一失。
从性能角度看,这种做法的优势很明显:页面结构得以优先渲染,用户能更快地看到内容。脚本的加载和执行被推迟,对首屏体验几乎没有影响。不过,这里有个平衡点需要注意:如果页面有需要立即交互的功能(比如一个导航菜单),就要评估脚本延迟执行是否会影响“功能就绪时间”。
立即学习“前端免费学习笔记(深入)”;
- 注意规范:不要写成
。虽然现代浏览器大多会“好心”地将其纠正到内,但这不符合HTML规范,可能会在CI流程或团队的代码检查工具中引发警告。 - 管理依赖:多个脚本应按依赖关系从上到下排列,避免再用手动延时或回调等方式去控制执行时机。
- 特殊情况:如果某个脚本必须提前加载(例如某些Polyfill),更推荐的做法是将其放在
中并加上defer属性,而不是硬塞到底部再用setTimeout等方式去等待DOM。
现代项目中 defer 是 head 里的唯一合理选择
除非你有非常特殊的需求,必须在DOM构建前就运行脚本(这种场景极少),否则在 中放置脚本时,defer 属性几乎是唯一合理的选择。它巧妙地解决了一个核心矛盾:既能让脚本(及其定义的全局变量或函数)在文档早期就被声明,又不会阻塞页面的渲染进程。
这里有个关键细节:defer 只对带有 src 的外部脚本生效,对内联脚本不起作用。它的执行时机是在DOM解析完成之后、DOMContentLoaded 事件触发之前。
async的适用场景:适合那些完全独立、没有依赖、且执行时机不重要的脚本,例如网站分析代码。但它既不保证执行顺序,也不等待DOM就绪。- 混合使用的顺序:当页面中同时存在
async和defer脚本时,async脚本会在加载完成后立即执行,因此其执行顺序可能早于defer脚本,无论它们在HTML中的书写顺序如何。 - 构建工具的默认行为:像Vite、Webpack这类现代构建工具,它们自动注入的运行时(runtime)脚本通常已经默认处理了
defer逻辑,开发者一般无需手动干预。
真正容易被忽略的细节
很多人以为“只要把脚本放在 前面就万事大吉了”,但实际开发中,还有一些隐蔽的坑等着我们。
首先,是脚本内部的写法问题。例如使用已被废弃的 document.write,或者在依赖的全局库(如jQuery)尚未加载完成时就急切地调用 $()。另外,在模块化开发中,误将ES Module当作普通脚本使用也需要警惕,因为ESM默认具有类似 async 的行为,即使放在body底部,其执行顺序也可能出乎意料。
其次,一个更隐蔽的问题来自构建流程。在使用服务端渲染(SSR)或静态站点生成器(如Hugo、Astro)时,最终输出的HTML中脚本的位置可能由构建工具自动决定并注入到 中。这时候,光看源代码文件的位置是没用的,必须去检查相应的构建配置才能找到根源。
