前端vue3实现图片懒加载

场景和指令用法

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

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

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

第一步:熟悉指令语法

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

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

我们可以使用useIntersectionObserver这个函数

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

javascript 复制代码
<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

javascript 复制代码
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
            }
        }
    )
  }
})

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

javascript 复制代码
<img v-img-lazy="item.picture"  alt="" />

页面效果

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

由上图可以看出当页面滑动到人气推荐时url全部都加载出来了

回顾核心步骤代码

================================补档优化==================================

问题1:逻辑书写位置不合理

问:懒加载指令的逻辑直接写到入口文件中,合理吗?

答:不合理,入口文件通常只做一些初始化的事情,不该包含太多的逻辑代码,可以通过插件的方法把懒加载指令封装为插件,main.js入口文件只需要负责注册插件即可

代码实现:

插件代码

javascript 复制代码
// 定义懒加载指令
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

javascript 复制代码
// 引入懒加载指令并且注册
import { lazyPlugin } from '@/directives'
app.use(lazyPlugin)

问题2:重复监听问题

uselntersectionObserver对于元素的监听是一直存在的,除非手动停止监听,存在内存浪费

解决思路:在监听的图片第一次完成加载之后就停止监听

代码实现:

javascript 复制代码
// 定义懒加载指令
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()
                        }
                    }
                )
            }
        })
    }
}
相关推荐
亿元程序员11 分钟前
明明直接用就可以了,非要在Creator里面写???
前端
wadesir34 分钟前
Nginx负载均衡代理协议详解(从零开始搭建高可用Web服务)
前端·nginx·负载均衡
秋氘渔36 分钟前
Vue 3 组合式写法:侦听器 watch 和 watchEffect 的区别及使用技巧
前端·javascript·vue.js·watch·watcheffect
光头程序员44 分钟前
vue学习笔记
vue.js·笔记·学习
想睡八个小时1 小时前
已包含的文件名 “a.vue“ 仅大小写与文件名 “A.vue“ 不同
前端·vscode
阿奇__1 小时前
element二次封装组件套餐 搜索组件 表格组件 弹窗组件
javascript·vue.js·elementui
The_era_achievs_hero1 小时前
Echarts
前端·javascript·echarts
czc1312 小时前
4K QPS 博客社区:CCBlog 全栈开源,Springboot项目实战,Docker一键部署
spring boot·redis·docker·开源·vue·rabbitmq
涔溪2 小时前
Vite 和 Webpack 这两款主流前端构建工具的核心区别,包括它们的设计理念、工作机制和实际使用体验上的差异。
前端·webpack·vite
0思必得02 小时前
[Web自动化] 开发者工具元素(Elements)面板
运维·前端·自动化·web自动化·开发者工具