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