前端vue3实现图片懒加载
场景和指令用法
场景:电商网站的首页通常会很长,用户不一定能访问到页面靠下面的图片,这类图片通过懒加载优化手段可以做到只有进入视口区域才发送图片请求
核心原理:图片进入视口才发送资源请求
首先:我们需要定义一个全局的指令,vue3官方的实现方法是这样的
第一步:熟悉指令语法
并且:还需要用到一个钩子函数
第二步:判断图片是否进入视口
我们可以使用useIntersectionObserver这个函数
以下是官方示例的使用方法:
import { useIntersectionObserver } from '@vueuse/core' const { stop } = useIntersectionObserver( target, ([entry], observerElement) => { targetIsVisible.value = entry?.isIntersecting || false }, )
target:需要监听的元素
isIntersecting:是一个布尔值 监听是否进入可视区
以下是完整代码实现
main.js
import { useIntersectionObserver } from '@vueuse/core' const app = createApp(App) // 定义全局指令 app.directive('img-lazy', { mounted(el, binding) { // el: 指令绑定的元素 // binding: binding.value 指令的绑定值 图片url console.log(el, binding.value); useIntersectionObserver( el, // 监听的元素 ([{ isIntersecting }]) => { // isIntersecting是一个布尔值 监听是否进入可视区 console.log(isIntersecting); if (isIntersecting) { // 图片进入可视区 设置图片的src el.src = binding.value } } ) } })
在需要懒加载的图片标签里使用这个即可
页面效果
由上图可以看出在刚进入页面时需要懒加载的图片没有加载出来
由上图可以看出当页面滑动到人气推荐时url全部都加载出来了
回顾核心步骤代码
================================补档优化==================================
问题1:逻辑书写位置不合理
问:懒加载指令的逻辑直接写到入口文件中,合理吗?
答:不合理,入口文件通常只做一些初始化的事情,不该包含太多的逻辑代码,可以通过插件的方法把懒加载指令封装为插件,main.js入口文件只需要负责注册插件即可
代码实现:
插件代码
// 定义懒加载指令 import { useIntersectionObserver } from '@vueuse/core' export const lazyPlugin = { install(app) { app.directive('img-lazy', { mounted(el, binding) { // el: 指令绑定的元素 // binding: binding.value 指令的绑定值 图片url console.log(el, binding.value); useIntersectionObserver( el, // 监听的元素 ([{ isIntersecting }]) => { // isIntersecting是一个布尔值 监听是否进入可视区 console.log(isIntersecting); if (isIntersecting) { // 图片进入可视区 设置图片的src el.src = binding.value } } ) } }) } }
main.js
// 引入懒加载指令并且注册 import { lazyPlugin } from '@/directives' app.use(lazyPlugin)
问题2:重复监听问题
uselntersectionObserver对于元素的监听是一直存在的,除非手动停止监听,存在内存浪费
解决思路:在监听的图片第一次完成加载之后就停止监听
代码实现:
// 定义懒加载指令 import { useIntersectionObserver } from '@vueuse/core' export const lazyPlugin = { install(app) { app.directive('img-lazy', { mounted(el, binding) { // el: 指令绑定的元素 // binding: binding.value 指令的绑定值 图片url console.log(el, binding.value); const {stop} = useIntersectionObserver( el, // 监听的元素 ([{ isIntersecting }]) => { // isIntersecting是一个布尔值 监听是否进入可视区 console.log(isIntersecting); if (isIntersecting) { // 图片进入可视区 设置图片的src el.src = binding.value stop() } } ) } }) } }
免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们。