高性能图片懒加载

图片懒加载(Lazy Loading)是一种网页优化技术,当图片需要显示时才加载,避免了不必要的流量消耗和服务器带宽浪费。

当页面包含大量图片时,浏览器会逐一发送 GET 请求获取每张图片的数据流,这会消耗用户的流量和服务器的带宽。无论是在 PC 端还是移动端,屏幕尺寸都是有限的,一次无法显示所有图片。因此,对于那些不在视口中的图片,我们无需立即加载,待用户滚动页面使图片进入可视区域时再进行加载。

基本原理

懒加载的概念非常简单:当页面上的图片进入可视区域时才加载它们。传统的懒加载方法通过监听滚动事件实现,但这种方法性能较差,需要频繁计算滚动位置。Intersection Observer API 则提供了一种更高效的解决方案。

懒加载的实现过程如下:

  1. 给图片设置一个占位图。
html 复制代码
<img src="占位图.png">
  1. 使用 data-src 属性保存真实图片的地址。
html 复制代码
<img src="placeholder.png" data-src="real-image.jpg">
  1. 当图片进入可视区域时,将 data-src 的值赋给 src 属性,从而加载真实图片。

传统方案:监听滚动事件

传统方法通过监听滚动事件来实现懒加载:

javascript 复制代码
window.addEventListener("scroll", () => {
  const imgs = document.querySelectorAll('img');
  imgs.forEach(img => {
    const rect = img.getBoundingClientRect();
    if (rect.top < document.body.clientHeight) {
      img.src = img.dataset.src;
    }
  });
});

这种方法需要频繁计算滚动位置,性能较差,还需使用防抖函数。

新方案:Intersection Observer API

Intersection Observer API 是浏览器原生提供的,用于异步检测目标元素与视口或父元素是否产生交叉。它的优势在于提高性能和简化代码实现。

实现代码

以下是使用 Intersection Observer 实现图片懒加载的完整代码:

javascript 复制代码
let observer = new IntersectionObserver((entries, observer) => {
  entries.forEach(entry => {
    if (entry.isIntersecting) {
      const img = entry.target;
      img.src = img.dataset.src;
      observer.unobserve(img);
    }
  });
}, {
  root: null,
  rootMargin: "0px",
  threshold: 0.1
});

document.querySelectorAll('img').forEach(img => observer.observe(img));

代码解析

  • IntersectionObserver 构造函数接受两个参数:回调函数和配置对象。
  • 回调函数处理所有进入视口的图片,将 data-src 赋值给 src,并取消观察。
  • rootnull 表示视口为根元素。
  • threshold0.1 表示当图片 10% 可见时触发回调。

效果

优势

  1. 高性能:只有在目标元素与视口交叉时才会触发回调。
  2. 简洁:不需要计算滚动距离,代码简洁明了。
  3. 兼容性好:现代浏览器都支持 Intersection Observer。
相关推荐
恶猫1 小时前
网页自动化模拟操作时,模拟真实按键触发事件【终级方案】
前端·javascript·自动化·vue·网页模拟
ZC跨境爬虫1 小时前
跟着 MDN 学 HTML day_2:(表单分组与高级输入控件实战)
前端·javascript·css·ui·html
千寻girling2 小时前
滑动窗口刷了快一个月(26天)了 , 还没有刷完. | 含(操作系统学什么的Java 后端)
java·开发语言·javascript·c++·人工智能·后端·python
一袋米扛几楼982 小时前
【报错问题】彻底解决 TypeScript 报错 TS2769: No overload matches this call (JWT 篇)
linux·javascript·typescript
牧码岛3 小时前
Web前端之JavaScrip中的Array、Object、Map和Set详解
前端·javascript·web·web前端
533_3 小时前
[pinia] vue3中监听pinia值的变化
前端·javascript·vue.js
AlenLi3 小时前
JavaScript - 相对实用的Axios二次封装
前端·javascript
一叶飘零晋3 小时前
【(二)Electron 使用之常用技巧】
javascript·electron·ecmascript
小小19924 小时前
vue 单页面请求
开发语言·前端·javascript
hhb_6184 小时前
JavaScript 本地存储与动态数据渲染实战案例
开发语言·javascript·ecmascript