脚本执行报document.querySelector返回null,是因为访问DOM时元素尚未解析;async会中断HTML解析立即执行,defer则确保DOM构建完成后再按序执行。
用错 async 或 defer,轻则 document.getElementById is not a function,重则业务逻辑全挂、埋点漏报、白屏时间翻倍——它们不是“加了就快”,而是“加对才稳”。
document.querySelector 返回 null?这是最典型的误用信号:脚本里访问了 DOM 元素,但元素还没解析到 HTML 中。常见于把本该 defer 的初始化代码写成 async,或干脆没加任何属性。
async 脚本下载完就中断 HTML 解析立即执行,此时 <body> 可能只解析了一半,#app 根本不存在defer 一定等到整个 HTML 解析完成、DOM 树构建完毕才执行,所以 document.body 和所有元素都可用<head> 里,很可能 DOM 还没开始建就执行了defer 为什么能保证多个脚本按顺序执行?因为浏览器明确把它“锁死”在两个节点之间:HTML 解析完成之后、DOMContentLoaded 事件触发之前。这个窗口期是确定的,且所有 defer 脚本共享同一个执行队列。
<script defer src="a.js"></script> 和 <script defer src="b.js"></script> 的书写顺序,跟文件大小、网络快慢无关lodash.js,再加载依赖它的 utils.js
defer 对内联脚本无效,<script defer>init();</script> 会立刻执行,不是延迟type="module" 脚本默认自带 defer 语义,手动加 defer 不会报错但也没效果async 真的“谁下完谁先执行”吗?是的,而且这个“先”可能发生在 DOM 构建中途,甚至在 <html> 标签刚打开时就执行——只要脚本下载够快。
立即学习“前端免费学习笔记(深入)”;
analytics.js(统计)、error-tracking.js(错误上报)、ads.js(广告),它们不操作 DOM、不依赖其他 JS、也不被其他 JS 依赖async 脚本之间绝对不能有调用关系,否则 utils.init() 很可能在 utils.js 加载完成前就运行async 和 defer 不能共存,浏览器会忽略 defer,只按 async 行为处理async 脚本,等于主动放弃服务端预渲染收益真正容易被忽略的点是:是否“依赖 DOM”和是否“被其他脚本依赖”必须同时判断——一个脚本既操作 document.body,又导出函数供后续脚本调用,那它只能走 defer;而哪怕只是发个请求、记个日志,只要不碰 DOM、不暴露 API,async 就是更干净的选择。