首屏白屏超1秒主因是关键渲染路径阻塞:外部CSS阻塞DOM构建,内联关键CSS可立即参与渲染树构建;非首屏HTML应“零渲染”(用template或动态插入);defer按序执行适合DOM依赖脚本,async无序执行仅适用于独立SDK;preload仅用于浏览器无法自动发现的关键资源(如首屏背景图、跨域字体),且需指定as和crossorigin属性。
首屏白屏时间超过1秒,大概率不是网络或后端问题,而是HTML结构和样式加载顺序没理清——关键渲染路径被阻塞了。
浏览器解析到外部 <link rel="stylesheet"> 时会暂停 DOM 构建,直到 CSSOM 完成;而内联 <style> 虽然增加 HTML 体积,但能立即参与渲染树构建,避免首次绘制延迟。
critters 或 mini-css-extract-plugin + critical 自动提取main.css,否则 HTML 变大反而拖慢传输,尤其在弱网下@import 或字体声明(@font-face),它们仍会触发额外请求并阻塞<style data-emotion> 并注入首屏 HTML所谓“非首屏”,是指用户滚动前不可见、且不参与初始布局计算的内容。强行渲染它们会延长 DOM 构建时间、增加内存占用,并可能触发无谓的重排。
Rendering > FPS Meter 和 Layers 面板观察首屏区域是否包含大量隐藏/折叠节点<template> 或纯 JS 动态插入(innerHTML 或 document.createElement)data-html 属性暂存字符串,切换时再解析挂载display: none 包裹非首屏区块——它仍会参与 CSSOM 计算和 DOM 构建,只是不绘制;真正要的是“不创建”两者都让脚本不阻塞 HTML 解析,但执行时机完全不同,选错会导致 DOM 操作失败或竞态问题。
立即学习“前端免费学习笔记(深入)”;
defer:脚本按出现顺序下载,等 DOM 解析完、DOMContentLoaded 前执行;适合依赖 DOM 结构的初始化逻辑(如 document.querySelector)async:脚本一下载完立刻执行,顺序不确定;只适合完全独立的脚本(如埋点、广告 SDK),且不能假设 document.body 已存在<script>...</script>)默认同步执行,若它不操作 DOM,可加 defer;若必须同步但又怕阻塞,就把它拆出来成外部文件再加 defer
type="module" 脚本——它天然具有 defer 行为,且支持 import 按需加载<link rel="preload"> 不是越多越好,滥用会抢占带宽、挤掉真正关键的资源。
<h1> 背景图、关键 Web 字体(font-display: optional 配合)、核心 icon font 文件preload 反而造成重复请求as="image" 和 fetchpriority="high"(Chrome 109+),否则可能被降级为低优先级crossorigin 属性,否则预加载失败且控制台报 CORS 错误最容易被忽略的一点:HTML 本身不是瓶颈,但它的组织方式决定了浏览器能否高效地调度后续所有资源。结构扁平、关键样式内联、非首屏内容“零渲染”、脚本加载策略匹配执行依赖——这四点卡准了,首屏性能优化才算真正落地。