当前位置: 首页 > article >正文

前端vue3实现图片懒加载

场景和指令用法

场景:电商网站的首页通常会很长,用户不一定能访问到页面靠下面的图片,这类图片通过懒加载优化手段可以做到只有进入视口区域才发送图片请求

核心原理:图片进入视口才发送资源请求

首先:我们需要定义一个全局的指令,vue3官方的实现方法是这样的

第一步:熟悉指令语法

并且:还需要用到一个钩子函数

第二步:判断图片是否进入视口

我们可以使用useIntersectionObserver这个函数

以下是官方示例的使用方法:

<script setup lang="ts">
import { useIntersectionObserver } from '@vueuse/core'const { stop } = useIntersectionObserver(target,([entry], observerElement) => {targetIsVisible.value = entry?.isIntersecting || false},
)
</script>

target:需要监听的元素

isIntersecting:是一个布尔值 监听是否进入可视区

以下是完整代码实现

main.js

import { useIntersectionObserver } from '@vueuse/core'
const app = createApp(App)
// 定义全局指令
app.directive('img-lazy', {mounted(el, binding) {// el: 指令绑定的元素// binding: binding.value 指令的绑定值 图片urlconsole.log(el, binding.value);useIntersectionObserver(el, // 监听的元素([{ isIntersecting }]) => { // isIntersecting是一个布尔值 监听是否进入可视区console.log(isIntersecting);       if (isIntersecting) {// 图片进入可视区 设置图片的srcel.src = binding.value}})}
})

在需要懒加载的图片标签里使用这个即可

<img v-img-lazy="item.picture"  alt="" />

页面效果

由上图可以看出在刚进入页面时需要懒加载的图片没有加载出来

由上图可以看出当页面滑动到人气推荐时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 指令的绑定值 图片urlconsole.log(el, binding.value);useIntersectionObserver(el, // 监听的元素([{ isIntersecting }]) => {// isIntersecting是一个布尔值 监听是否进入可视区console.log(isIntersecting);if (isIntersecting) {// 图片进入可视区 设置图片的srcel.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 指令的绑定值 图片urlconsole.log(el, binding.value);const {stop} =  useIntersectionObserver(el, // 监听的元素([{ isIntersecting }]) => {// isIntersecting是一个布尔值 监听是否进入可视区console.log(isIntersecting);if (isIntersecting) {// 图片进入可视区 设置图片的srcel.src = binding.valuestop()}})}})}
}

http://www.lryc.cn/news/2392538.html

相关文章:

  • 网站每天几点更新,更新频率是否影响网站收录
  • 主流Markdown编辑器的综合评测与推荐
  • 计算机网络-MPLS VPN应用场景与组网
  • AugmentFree:解除 AugmentCode 限制的终极方案 如何快速清理vscode和AugmentCode缓存—windows端
  • WPF【11_7】WPF实战-重构与美化(ViewModel的嵌套与分解、海量数据不要Join)
  • Linux 的编辑器--vim
  • Oracle 慢sql排查
  • [Protobuf] 快速上手:安全高效的序列化指南
  • uniapp开发企业微信小程序时 wx.qy.login 在uniapp中使用的时候,需要导包吗?
  • 如何将通话记录从Android传输到Android
  • Word 目录自动换行后错位与页码对齐问题解决教程
  • 数据结构第4章 栈、队列和数组 (竟成)
  • removeIf() 方法,结合 Lambda 表达式
  • 汽车售后诊断数据流详细分析
  • 2025年渗透测试面试题总结-匿名[校招]安全研究员(SAST方向)(题目+回答)
  • Unity 游戏优化(持续更新中...)
  • LlamaFactory——如何使用魔改后的模型
  • 【前端】【css预处理器】Sass与Less全面对比与构建对应知识体系
  • 【请关注】关于VC++实现使用Redis不同方法,有效达到 Redis 性能优化、防击穿
  • 【加密算法】
  • 20250529-C#知识:索引器
  • 【笔记】suna部署之获取 Tavily API key
  • 06-Web后端基础(java操作数据库)
  • JavaScript性能优化实战的技术文-——仙盟创梦IDE
  • GitHub Copilot 使用手册与原理解析
  • vllm 2080TI ubuntu环境安装
  • 【C语言】指针详解(接)
  • 【C】箭头运算符
  • HTTP Accept简介
  • 什么是单片机?