前端图片懒加载布局抖动治理完整流程:占位比例、按需加载和 CLS 复查
图片懒加载能减少首屏资源请求,但如果只把图片延后加载、没有给图片区域预留尺寸,页面反而会出现明显跳动。用户刚准备点击按钮,图片突然撑开页面,按钮位置变了,这就是典型的布局偏移问题。
这篇文章按完整工作流处理:先界定目标,再找出布局抖动原因,接着用固定比例占位、IntersectionObserver 按需加载和加载状态管理解决问题,最后用 CLS 指标复查。
目录
- 目标和边界:懒加载不等于布局稳定
- 全流程总览:从图片抖动到 CLS 稳定
- 阶段 1:先给图片区域预留比例
- 阶段 2:进入视口附近再发起加载
- 阶段 3:加载完成后稳定替换占位
- 推荐工作流:懒加载、占位和指标复查
- 容易踩坑的地方
- 速查表
目标和边界:懒加载不等于布局稳定
本次目标不是单纯“少加载几张图”,而是在减少首屏请求的同时保持布局稳定。一个合格的图片懒加载方案至少要满足三个条件:
- 图片加载前,列表卡片已经有稳定高度。
- 图片接近视口时再开始请求,避免过早加载。
- 图片替换占位时不改变卡片尺寸。
本文示例适合商品列表、文章封面、瀑布流前的普通列表、用户头像和卡片封面。复杂瀑布流还要额外处理列高计算,这里先不展开。
全流程总览:从图片抖动到 CLS 稳定
先看整体链路。布局抖动通常不是懒加载本身造成的,而是“图片尺寸未知”。浏览器第一次排版时不知道图片要占多高,图片回来后重新撑开内容,就产生了位移。

所以治理顺序应该是:先预留空间,再做懒加载,最后复查指标。顺序反了,就会变成“请求少了,但页面还是跳”。
阶段 1:先给图片区域预留比例
目标
让图片没加载出来时,卡片高度已经确定。这样图片回来后只是替换内容,不会把下面的文字、按钮或下一张卡片挤下去。
关键动作
推荐用固定比例容器包住图片:
.thumb {
width: 100%;
aspect-ratio: 16 / 9;
background: #f2f4f7;
overflow: hidden;
}
.lazy-img {
width: 100%;
height: 100%;
object-fit: cover;
opacity: 0;
transition: opacity .2s ease;
}
.lazy-img.is-loaded {
opacity: 1;
}
检查点
| 检查项 | 期望结果 |
|---|---|
| 图片未加载 | 卡片高度已经稳定 |
| 图片加载完成 | 只发生透明度变化,不推动内容 |
| 窗口宽度变化 | 图片区域按比例缩放 |
阶段 2:进入视口附近再发起加载
目标
图片不用一开始全部请求,但也不能等到完全进入视口才加载,否则用户滚动时可能看到空白。通常可以提前一段距离触发。
关键动作
const images = document.querySelectorAll('.lazy-img[data-src]');
const observer = new IntersectionObserver((entries) => {
entries.forEach((entry) => {
if (!entry.isIntersecting) return;
const img = entry.target;
img.src = img.dataset.src;
observer.unobserve(img);
});
}, {
rootMargin: '200px 0px',
threshold: 0.01
});
images.forEach((img) => observer.observe(img));
检查点
打开网络面板后,首屏外图片不应该一开始全部请求;向下滚动时,接近视口的图片才开始加载。
阶段 3:加载完成后稳定替换占位
目标
图片请求成功后再显示,失败时也要保留原来的占位区域,不能让卡片高度塌掉。
关键动作
document.querySelectorAll('.lazy-img').forEach((img) => {
img.addEventListener('load', () => {
img.classList.add('is-loaded');
});
img.addEventListener('error', () => {
img.classList.add('is-error');
img.alt = '图片加载失败';
});
});
失败状态也应该在同一个固定比例容器内展示,例如显示浅色背景、错误图标或简短提示。核心原则是:成功和失败都不改变容器尺寸。
推荐工作流:懒加载、占位和指标复查
落地时可以按下面的顺序做,不要一上来只写观察器代码。

- 先统计页面里会延迟加载的图片区域。
- 为每类图片确定固定比例,例如 16:9、4:3、1:1。
- 用 CSS 预留空间,并准备骨架背景。
- 用
IntersectionObserver在接近视口时设置真实图片地址。 - 监听加载成功和失败状态,保证替换过程不改尺寸。
- 用 Performance 面板或线上监控复查 CLS。
容易踩坑的地方
- 只加 loading="lazy":原生懒加载能延后请求,但不能自动帮你预留空间。
- 只写 width 不写高度:宽度稳定不代表高度稳定,列表仍然可能被图片撑开。
- 骨架屏高度和真实图片不一致:替换时依然会跳,骨架区域要和图片容器一致。
- 图片失败时移除节点:失败也要保留占位,否则页面高度会突然变化。
- 只在本地看一遍:慢网速、缓存关闭、移动端宽度下更容易暴露布局抖动。
速查表
| 阶段 | 动作 | 检查点 |
|---|---|---|
| 预留空间 | 使用 aspect-ratio 或固定尺寸容器 |
图片未加载时高度稳定 |
| 懒加载 | 接近视口再设置真实图片地址 | 首屏外图片不会过早请求 |
| 替换显示 | 加载成功后添加显示状态 | 内容不被挤压 |
| 失败兜底 | 保留占位区域并提示失败 | 容器不塌陷 |
| 指标复查 | 检查 CLS 或线上性能监控 | 页面位移进入安全范围 |
总结一下:图片懒加载不是只把请求延后,而是要和布局占位一起设计。先把尺寸边界定住,再按需加载图片,最后复查 CLS,用户看到的页面才会又快又稳。
Java 空指针异常排查实战:从日志定位到入口校验和安全映射
- 上一篇
- Java 空指针异常排查实战:从日志定位到入口校验和安全映射
- 下一篇
- DBeaver 连接 MySQL 实战:新建连接、测试连通和运行 SQL
-
- 文章 · 前端 | 21小时前 | 工程化 · 前端 · javascript · css · 弹窗 · 前端 z-index 遮罩层 stacking context Portal 弹窗层级
- 前端弹窗层级治理工作流:从 z-index 混乱到 Portal 容器规范
- 350浏览 收藏
-
- 文章 · 前端 | 21小时前 | 前端 · javascript · URL参数 · 列表筛选 · 页面状态 · 前端 筛选条件 列表页 history.replaceState URLSearchParams 刷新还原
- 前端筛选条件刷新后丢失怎么办:从内存状态到 URL 参数一步步排查
- 348浏览 收藏
-
- 文章 · 前端 | 23小时前 | 前端 · 性能优化 · 路由 · javascript · 前端 用户体验 滚动位置 路由缓存 scrollRestoration
- 前端详情页返回列表丢失滚动位置怎么办:从复现到恢复一步步排查
- 458浏览 收藏
-
- 文章 · 前端 | 3天前 | 前端 · javascript · sourcemap · 错误监控 · 线上排查 · 前端 错误监控 告警 onerror sourcemap unhandledrejection
- 前端错误监控实战:onerror、unhandledrejection 和 sourcemap 定位问题
- 331浏览 收藏
-
- 文章 · 前端 | 3天前 | 前端 · javascript · 缓存治理 · localStorage · Web性能 · 前端 本地缓存 localStorage 过期时间 版本迁移 异常兜底
- 前端 localStorage 缓存治理实战:过期时间、版本号和异常兜底
- 480浏览 收藏
-
- 文章 · 前端 | 3天前 | 前端 · 性能优化 · javascript · 图片优化 · IntersectionObserver · 前端 性能优化 图片懒加载 IntersectionObserver Web性能 首屏优化
- 前端图片懒加载实战:用 IntersectionObserver 降低首屏压力
- 184浏览 收藏
-
- 文章 · 前端 | 3天前 | 前端 · 性能优化 · javascript · fetch · 前端 搜索优化 Fetch AbortController 请求竞态
- 前端搜索竞态治理实战:用 AbortController 取消过期请求
- 178浏览 收藏
-
- 文章 · 前端 | 4天前 |
- 前端长任务治理实战:用 PerformanceObserver 找出页面卡顿源头
- 423浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 6次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 15次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 24次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 166次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 168次使用
-
- 搞一个自娱自乐的博客(二) 架构搭建
- 2023-02-16 244浏览
-
- 搞一个自娱自乐的博客(四) 友链
- 2023-02-24 322浏览
-
- Go语言对前端领域的入侵WebAssembly运行原理
- 2022-12-31 130浏览
-
- go zero微服务实战性能优化极致秒杀
- 2022-12-27 207浏览
-
- 当自己的项目被开源中国推荐后是什么感觉
- 2023-02-25 205浏览

