许多人误以为在页面中放入一个 ,加载进度就会从 0 开始自动走到 100——这个想法基本不成立。先明确一个关键结论:原生 标签仅仅是一个空壳,浏览器绝不会主动将 HTML 解析、CSS 下载、图片加载等过程映射到它的 value 上。你所看到的那个静止不动的 0%,正是它最真实的状态。
想让它动起来,必须亲自动手:跟踪资源加载、计算比例、手动更新 value。实际开发中,有几个容易踩坑的细节需要特别注意。

为什么 什么也不发生
简单来说, 只识别两个数字属性:value 和 max,而且必须手动赋值。更关键的是,setAttribute("value", "30") 这种写法是无效的——它仅修改 HTML attribute,不会触发 DOM 属性的更新和渲染。正确的做法是直接操作 DOM 属性:el.value = 30。
在实际项目中,常见的翻车点有以下几种:第一,没有设置 max="100",依赖默认值,结果后端返回小数(比如 0.73)时直接赋值,进度条直接跑飞或卡死;第二,在 img.onload 回调里写 progress.value = count / total * 100,却忘了加 Math.round(),小数精度问题导致进度停在 99.99999999999999% 的情况很常见;第三,有人使用 document.readyState 来判断,但它只有 "loading"、"interactive"、"complete" 三种状态,根本无法线性映射到 0–100%。
用 PerformanceObserver 捕获真实资源加载占比
从技术实现角度看,PerformanceObserver 是目前最接近“真实进度”的方案,特别适合 Chrome 80+ 及现代 Edge。它能捕捉到每个资源的 startTime 和 duration,结合 na vigationStart 可以估算出相对耗时占比。
不过,这个方案有几个关键实操点需要注意。首先,必须在 最早的位置初始化,否则会错过首屏 CSS、字体等关键资源。其次,要记得过滤掉 initiatorType === "xmlhttprequest" 的条目——那些是 API 请求,不属于页面初始加载的资源。真正需要统计的是 entry.initiatorType === "na vigation"、"link"、"script" 这一类,也就是 HTML、CSS、同步 JS、内联脚本。另外,动态插入的资源,比如懒加载的 img、import() 等,需要单独监听 load 事件并计入总完成数。
示例片段可以参考这段(已经简化处理):
const observer = new PerformanceObserver((list) => {
const entries = list.getEntries().filter(e =>
e.entryType === "resource" &&
["na vigation", "link", "script"].includes(e.initiatorType)
);
const totalDuration = entries.reduce((sum, e) => sum + e.duration, 0);
const loadedDuration = entries.filter(e => e.duration > 0)
.reduce((sum, e) => sum + e.duration, 0);
const percent = Math.round((loadedDuration / Math.max(totalDuration, 1)) * 100);
progress.value = Math.min(percent, 100);
});
observer.observe({ entryTypes: ["resource"] });
用资源数量比例做轻量 fallback 方案
如果目标环境不支持 PerformanceObserver(比如旧版 Safari),或者页面资源结构比较简单——例如以图片为主,可以退而求其次使用数量计数法。原理很简单:预先列出关键资源,监听它们的 load 或 error 事件。
边界条件需要仔细处理:第一,只统计 document.querySelectorAll('img, link[rel="stylesheet"], script:not([async]):not([defer])'),忽略 async/defer 脚本,因为它们不阻塞解析,不算进“页面加载”主路径。第二,link[rel="preload"] 需要单独加入队列,否则进度会提前结束。第三,图片的 srcset 或 picture 中的多个源,按实际加载的那个算 1 个,避免重复计数。此外,务必加超时兜底,比如 8 秒后强制设为 100%,防止某个资源挂起导致进度条卡死。
transform: scaleX() 比 width 更稳的动画写法
用 JS 频繁修改 width 会触发重排,尤其在低端安卓机上容易掉帧;transform: scaleX(n) 是合成层操作,性能更优。但兼容细节需要注意:iOS 15 之前对 scaleX(0) 有渲染 bug,起始值建议设为 scaleX(0.001)。同时,不要在 load 回调里直接改样式,先用变量缓存当前百分比,再用 requestAnimationFrame 统一提交。
话说回来,如果使用 ,它的内部 track 渲染由浏览器控制,不需要手动加 transform;但自定义 时,必须走 transform 路线。
说到底,真正难的不是让进度条动起来,而是定义清楚「什么才算加载完成」——是 HTML 解析完?首屏 CSS 生效?还是所有 img 和字体都到位?不同业务目标,监控粒度和兜底策略完全不同。漏掉一个异步资源,或者误判一个 defer 脚本,百分比就会失真。这才是实践中最值得警惕的地方。
