图片懒加载实现方式

2024-08-07 09:49:13 浏览数 (3)

图片作为前端开发中不可或缺的元素,其加载速度对用户体验有着重要影响。然而,大量的图片加载不仅会消耗用户流量,还会导致页面加载缓慢,影响用户体验。为了解决这个问题,图片懒加载技术应运而生

图片懒加载(Lazy Loading)是一种优化网页性能的技术,它通过延迟加载图片,即在图片即将进入可视区域时才开始加载,从而减少页面初始加载时间,提高页面响应速度。

图片懒加载的原理

图片懒加载的实现原理主要基于以下几个关键点:

  • 滚动事件监听: 图片懒加载的核心是通过监听浏览器的滚动事件(scroll事件)。当用户滚动页面时,会触发这个事件。
  • 可视区域检测: 在滚动事件触发时,需要检测每个图片元素是否已经进入或即将进入浏览器的可视区域。这通常通过以下几种方法实现:
    • 基于Element的getBoundingClientRect()方法:这个方法可以获取元素的位置和尺寸信息,通过计算元素相对于视口的位置,可以判断元素是否在可视区域内。
    • Intersection Observer API:这是一个现代的API,可以异步观察目标元素与其祖先元素或顶级文档视口的交叉状态。它提供了更加简洁和高效的方式来监听元素是否进入可视区域。
    • 条件加载: 当检测到图片即将进入可视区域时,才开始加载这张图片。这样可以避免在页面初始加载时加载所有图片,从而减少初始加载时间和内存消耗。
    • 资源替换: 在图片检测到即将进入可视区域时,使用JavaScript动态地将图片的src属性设置为实际的图片URL。如果使用占位符(如低分辨率图片或单色图片),则在加载完成后将其替换为实际的图片资源。

实现方式

利用滚动事件监听 getBoundingClientRect

原理: 图片dom 预先不设置src属性值,而新增自定义属性 wait-render值为true,初始化 预渲染3张,监听dom滚动事件,当到达可视范围域,开始加载图片

设置图片的 src 属性为实际图片 URL,并删除wait-render属性

使用vue3 实现,注意要点

1.滚动事件可用 @scroll监听

2.循环中的dom用ref的方式获取可以利用ref绑定一个方法,然后插入到数组中备用

3.初始化和后续监听中有重复逻辑 抽离公用设置图片setImg,参数为方法返回满足条件

代码语言:javascript复制
<template>
  <div ref="scrollContainer" @scroll="lazyLoadImages" class="image-container">
    <img :ref="getImg" v-for="(image, index) in images" :key="image" :wait-render="true" alt="图片" />
  </div>
</template>

<script setup>
import {  ref } from 'vue';

const scrollContainer = ref(null);
// 存储图片数据
const images = ref([
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/a1ec08fa513d2697e542494057fbb2fb4316d81e.jpg",
  "http://c.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/30adcbef76094b36de8a2fe5a1cc7cd98d109d99.jpg",
  "http://h.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/7c1ed21b0ef41bd5f2c2a9e953da81cb39db3d1d.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/55e736d12f2eb938d5277fd5d0628535e5dd6f4a.jpg",
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4e4a20a4462309f7e41f5cfe760e0cf3d6cad6ee.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/9d82d158ccbf6c81b94575cfb93eb13533fa40a2.jpg",
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4bed2e738bd4b31c1badd5a685d6277f9e2ff81e.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/0d338744ebf81a4c87a3add4d52a6059252da61e.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/f2deb48f8c5494ee5080c8142ff5e0fe99257e19.jpg",
  "http://f.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4034970a304e251f503521f5a586c9177e3e53f9.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/279759ee3d6d55fbb3586c0168224f4a20a4dd7e.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/e824b899a9014c087eb617650e7b02087af4f464.jpg",
  "http://c.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/9c16fdfaaf51f3de1e296fa390eef01f3b29795a.jpg",
  "http://d.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/b58f8c5494eef01f119945cbe2fe9925bc317d2a.jpg",
  "http://h.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/902397dda144ad340668b847d4a20cf430ad851e.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/359b033b5bb5c9ea5c0e3c23d139b6003bf3b374.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/8d5494eef01f3a292d2472199d25bc315d607c7c.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/e824b899a9014c08878b2c4c0e7b02087af4f4a3.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/6d81800a19d8bc3e770bd00d868ba61ea9d345f2.jpg",
]);
let refImgs = ref([])
//遍历获取元素,然后放在一个数组里
function getImg (el) {
  console.log('el', el)
  if (el) {
    refImgs.value.push(el)
  }
}
// 懒加载函数
const lazyLoadImages = () => {
  // 获取可视区域的高度
  const windowHeight = window.innerHeight;
  // 遍历所有图片
  setImg((image, index) => {
    let img = refImgs.value[index]
    if(!img.getAttribute('wait-render')){
      return false
    }
    // 获取图片距离视口顶部的距离
    const imgTop = img.getBoundingClientRect().top;
    if (imgTop >= windowHeight){
      return false 
    }
    // 到达可视范围域,开始加载图片
    // 设置图片的 src 属性为实际图片 URL
    return true
  })
};

const setImg = (func = () => { }) => {
  images.value.forEach((image, index) => {
    if (func(image, index)) {
      const img = refImgs.value[index]
      img.src = image;
      // 加载完成后移除 wait-render 属性
      img.removeAttribute('wait-render');
    }
  })
}
onMounted(() => {
  setImg((image, index) => {
    return index < 3
  })
});
onUnmounted(() => {
});
</script>

<style>
.image-container {
  width: 100%;
  height: 100vh;
  overflow: hidden;
  overflow-y: scroll;
}

img {
  width: 100%;
  display: block;
  margin-bottom: 20px;
}
</style>

效果展示

Intersection Observer

从上图中滚动到加载图片的效果分析,看起来并不怎么丝滑,加载时机也不是很准确,以下是优化分析

  • 1.当前代码中,图片加载是按顺序进行的,这可能导致滚动到页面的底部时,页面加载速度变慢。可以考虑使用异步加载或分批加载图片,以提高用户体验。使用Intersection Observer API代替手动计算图片位置,这样可以更精确地控制图片加载时机。
  • 2.refImgs数组用于存储图片DOM元素的引用,但这个数组并不需要响应式。可以将它改为普通的JavaScript数组。(这个确实,所以考虑连这个refImgs变量声明都省了,直接用父级节点来获取子集scrollContainer.children)

修改之前 先了解下 Intersection Observer这个api

Intersection Observer API

它一个现代浏览器的API,用于异步观察目标元素与其祖先元素或顶级文档视窗(viewport)交叉状态的变化。这个API允许开发者在不使用轮询(polling)的情况下,高效地检测元素是否进入、离开或部分进入视窗。

1.基本概念
  • 目标元素(Target Element):想要观察的元素。
  • 祖先元素(Ancestor Element):目标元素的父元素或更上层的元素,或者是整个文档。
  • 视窗(Viewport):浏览器窗口的可见部分。
2.事件
  • 当目标元素与视窗交叉的状态发生变化时,会触发回调函数。以下是可能发生的事件:
  • 进入视窗(Enter the viewport):目标元素首次进入视窗。
  • 离开视窗(Leave the viewport):目标元素完全离开视窗。
  • 部分进入视窗(Partially enter the viewport):目标元素部分进入视窗。
3.使用方法

以下是一个简单的Intersection Observer API的使用示例:

代码语言:javascript复制
// 创建一个Intersection Observer实例
let observer = new IntersectionObserver((entries, observer) => {
  entries.forEach(entry => {
    // entry.isIntersecting为true表示目标元素进入了视窗
    if (entry.isIntersecting) {
      console.log('元素已进入视窗');
      // 可以在这里执行代码,例如加载图片、显示元素等
    } else {
      console.log('元素已离开视窗');
      // 可以在这里执行代码,例如隐藏元素等
    }
  });
}, {
  // 在视窗中至少有50%的目标元素可见时触发回调
  threshold: 0.5
});

// 开始观察目标元素
observer.observe(document.getElementById('target-element'));

// 如果需要停止观察,可以调用
// observer.unobserve(document.getElementById('target-element'));

开始改造

下面利用 Intersection Observer改造后的完整代码

注意 图片要给个默认高度来撑开父级元素,否则初始化的时候图 都堆积在一起, 所以Intersection Observer会判定在可视窗口内的img 造成过度加载。就达不到想要的效果了

代码语言:javascript复制
<template>
  <div ref="scrollContainer" class="image-container">
    <img v-for="(image, index) in images" :key="index" :data-src="image" alt="图片" />
  </div>
</template>

<script setup>
import { ref } from 'vue';

const images = ref([
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/a1ec08fa513d2697e542494057fbb2fb4316d81e.jpg",
  "http://c.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/30adcbef76094b36de8a2fe5a1cc7cd98d109d99.jpg",
  "http://h.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/7c1ed21b0ef41bd5f2c2a9e953da81cb39db3d1d.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/55e736d12f2eb938d5277fd5d0628535e5dd6f4a.jpg",
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4e4a20a4462309f7e41f5cfe760e0cf3d6cad6ee.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/9d82d158ccbf6c81b94575cfb93eb13533fa40a2.jpg",
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4bed2e738bd4b31c1badd5a685d6277f9e2ff81e.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/0d338744ebf81a4c87a3add4d52a6059252da61e.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/f2deb48f8c5494ee5080c8142ff5e0fe99257e19.jpg",
  "http://f.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4034970a304e251f503521f5a586c9177e3e53f9.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/279759ee3d6d55fbb3586c0168224f4a20a4dd7e.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/e824b899a9014c087eb617650e7b02087af4f464.jpg",
  "http://c.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/9c16fdfaaf51f3de1e296fa390eef01f3b29795a.jpg",
  "http://d.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/b58f8c5494eef01f119945cbe2fe9925bc317d2a.jpg",
  "http://h.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/902397dda144ad340668b847d4a20cf430ad851e.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/359b033b5bb5c9ea5c0e3c23d139b6003bf3b374.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/8d5494eef01f3a292d2472199d25bc315d607c7c.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/e824b899a9014c08878b2c4c0e7b02087af4f4a3.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/6d81800a19d8bc3e770bd00d868ba61ea9d345f2.jpg",
]);
const scrollContainer = ref(null);

// Intersection Observer 用于懒加载图片
let observer = null;


function loadImage(imageElement) {
  // 使用data-src属性存储真实的图片地址
  const src = imageElement.dataset.src;
  if (src) {
    imageElement.src = src;
    imageElement.removeAttribute('data-src');
  }
}

/**
 * @param {Function} entries 一个数组,包含每个被观察元素的交叉信息。
 * @param {Number} observer IntersectionObserver 实例本身。
 */
const observerCallback = (entries, observer) => {
  entries.forEach(entry => {
    if (entry.isIntersecting) {
      loadImage(entry.target);
      observer.unobserve(entry.target);
    }
  });
};

onMounted(() => {
  // 创建IntersectionObserver
  observer = new IntersectionObserver(observerCallback, {
    root: scrollContainer.value, // 观察目标元素
    threshold: 0.1 // 当10%的元素可见时触发回调 observerCallback
  });
  // 循环图片dom 开始观察每个图片dom是否在可视窗口 10%可见处
  images.value.forEach((image, index) => {
    const imgElement = scrollContainer.value.children[index];
    observer.observe(imgElement);   // 开始观察
  });
});

onUnmounted(() => {
  if (observer) {
    observer.disconnect(); // 停止观察所有元素
  }
});

</script>

<style>
.image-container {
  width: 100%;
  height: 100vh;
  overflow: hidden;
  overflow-y: scroll;
}

img {
  width: 100%;
  height: 500px;
  display: block;
  margin-bottom: 20px;
  object-fit: cover;
}
</style>

效果图

这样看起来就丝滑多了,加载时机也很准确,但每次使用 都要写这么多逻辑是不是很繁琐,用起来也不是很方便,能不能封装起来,让使用更加简洁和减少代码量书写呢,其实可以,而且不用重复造轮子,已经有成熟的组件库了,下面说一下 vue3-lazyload

vue3-lazyload

vue3-lazyload 是一个基于 Vue 3 的懒加载组件,它允许你延迟加载图片、视频或其他资源,直到它们接近或进入视口(用户可见的区域)。

这个组件库 能实现和 Intersection Observer一样的效果,而且使用非常方便,并且已经内置了加载逻辑,让代码看起来简洁很多

安装 vue3-lazyload

代码语言:shell复制
npm install vue3-lazyload

全局注册

代码语言:javascript复制
<!--main.js-->
...
import Lazyload from "vue3-lazyload";

const app = createApp(App)
//注册插件
app.use(Lazyload, {
  loading: "@/assets/img/default.png",//可以指定加载中的图像
  error: "@/assets/img/error.png",//可以指定加载失败的图像
});
...

使用完整案例

代码语言:javascript复制
<template>
  <div class="image-container">
    <template v-for="(url, index) in images" :key="index">
      <img class="img" v-lazy="url" alt="图片" />
    </template>
  </div>
</template>

<script setup>
import { ref } from 'vue';

const images = ref([
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/a1ec08fa513d2697e542494057fbb2fb4316d81e.jpg",
  "http://c.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/30adcbef76094b36de8a2fe5a1cc7cd98d109d99.jpg",
  "http://h.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/7c1ed21b0ef41bd5f2c2a9e953da81cb39db3d1d.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/55e736d12f2eb938d5277fd5d0628535e5dd6f4a.jpg",
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4e4a20a4462309f7e41f5cfe760e0cf3d6cad6ee.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/9d82d158ccbf6c81b94575cfb93eb13533fa40a2.jpg",
  "http://e.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4bed2e738bd4b31c1badd5a685d6277f9e2ff81e.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/0d338744ebf81a4c87a3add4d52a6059252da61e.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/f2deb48f8c5494ee5080c8142ff5e0fe99257e19.jpg",
  "http://f.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/4034970a304e251f503521f5a586c9177e3e53f9.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/279759ee3d6d55fbb3586c0168224f4a20a4dd7e.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/e824b899a9014c087eb617650e7b02087af4f464.jpg",
  "http://c.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/9c16fdfaaf51f3de1e296fa390eef01f3b29795a.jpg",
  "http://d.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/b58f8c5494eef01f119945cbe2fe9925bc317d2a.jpg",
  "http://h.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/902397dda144ad340668b847d4a20cf430ad851e.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/359b033b5bb5c9ea5c0e3c23d139b6003bf3b374.jpg",
  "http://a.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/8d5494eef01f3a292d2472199d25bc315d607c7c.jpg",
  "http://b.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/e824b899a9014c08878b2c4c0e7b02087af4f4a3.jpg",
  "http://g.hiphotos.baidu.com/imagehttps://img.yuanmabao.com/zijie/pic/item/6d81800a19d8bc3e770bd00d868ba61ea9d345f2.jpg",
]);


</script>

<style>
.image-container {
  width: 100%;
  overflow: hidden;
  overflow-y: scroll;
}

.img {
  width: 100%;
  height: 500px;
  display: block;
  margin-bottom: 20px;
  object-fit: cover;
}
</style>

1 人点赞