前端vue3實(shí)現(xiàn)圖片懶加載的完整步驟記錄
場(chǎng)景和指令用法
場(chǎng)景:電商網(wǎng)站的首頁(yè)通常會(huì)很長(zhǎng),用戶不一定能訪問到頁(yè)面靠下面的圖片,這類圖片通過懶加載優(yōu)化手段可以做到只有進(jìn)入視口區(qū)域才發(fā)送圖片請(qǐng)求
核心原理:圖片進(jìn)入視口才發(fā)送資源請(qǐng)求

首先:我們需要定義一個(gè)全局的指令,vue3官方的實(shí)現(xiàn)方法是這樣的
第一步:熟悉指令語(yǔ)法

并且:還需要用到一個(gè)鉤子函數(shù)

第二步:判斷圖片是否進(jìn)入視口
我們可以使用useIntersectionObserver這個(gè)函數(shù)
以下是官方示例的使用方法:
<script setup lang="ts">
import { useIntersectionObserver } from '@vueuse/core'
const { stop } = useIntersectionObserver(
target,
([entry], observerElement) => {
targetIsVisible.value = entry?.isIntersecting || false
},
)
</script>target:需要監(jiān)聽的元素
isIntersecting:是一個(gè)布爾值 監(jiān)聽是否進(jìn)入可視區(qū)
以下是完整代碼實(shí)現(xiàn)
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, // 監(jiān)聽的元素
([{ isIntersecting }]) => {
// isIntersecting是一個(gè)布爾值 監(jiān)聽是否進(jìn)入可視區(qū)
console.log(isIntersecting);
if (isIntersecting) {
// 圖片進(jìn)入可視區(qū) 設(shè)置圖片的src
el.src = binding.value
}
}
)
}
})在需要懶加載的圖片標(biāo)簽里使用這個(gè)即可
<img v-img-lazy="item.picture" alt="" />
頁(yè)面效果

由上圖可以看出在剛進(jìn)入頁(yè)面時(shí)需要懶加載的圖片沒有加載出來

由上圖可以看出當(dāng)頁(yè)面滑動(dòng)到人氣推薦時(shí)url全部都加載出來了
回顧核心步驟代碼

================================補(bǔ)檔優(yōu)化==================================
問題1:邏輯書寫位置不合理問:懶加載指令的邏輯直接寫到入口文件中,合理嗎?
答:不合理,入口文件通常只做一些初始化的事情,不該包含太多的邏輯代碼,可以通過插件的方法把懶加載指令封裝為插件,main.js入口文件只需要負(fù)責(zé)注冊(cè)插件即可

代碼實(shí)現(xiàn):
插件代碼

// 定義懶加載指令
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, // 監(jiān)聽的元素
([{ isIntersecting }]) => {
// isIntersecting是一個(gè)布爾值 監(jiān)聽是否進(jìn)入可視區(qū)
console.log(isIntersecting);
if (isIntersecting) {
// 圖片進(jìn)入可視區(qū) 設(shè)置圖片的src
el.src = binding.value
}
}
)
}
})
}
}
main.js
// 引入懶加載指令并且注冊(cè)
import { lazyPlugin } from '@/directives'
app.use(lazyPlugin)問題2:重復(fù)監(jiān)聽問題uselntersectionObserver對(duì)于元素的監(jiān)聽是一直存在的,除非手動(dòng)停止監(jiān)聽,存在內(nèi)存浪費(fèi)
解決思路:在監(jiān)聽的圖片第一次完成加載之后就停止監(jiān)聽
代碼實(shí)現(xiàn):
// 定義懶加載指令
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, // 監(jiān)聽的元素
([{ isIntersecting }]) => {
// isIntersecting是一個(gè)布爾值 監(jiān)聽是否進(jìn)入可視區(qū)
console.log(isIntersecting);
if (isIntersecting) {
// 圖片進(jìn)入可視區(qū) 設(shè)置圖片的src
el.src = binding.value
stop()
}
}
)
}
})
}
}總結(jié)
到此這篇關(guān)于前端vue3實(shí)現(xiàn)圖片懶加載的文章就介紹到這了,更多相關(guān)前端vue3圖片懶加載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳細(xì)講一講vue3下會(huì)造成響應(yīng)式丟失的情況
vue3開發(fā)過程中,綁定的響應(yīng)式數(shù)據(jù)失去了響應(yīng)式,如何解決問題呢,下面這篇文章主要給大家介紹了關(guān)于vue3下會(huì)造成響應(yīng)式丟失的情況,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-06-06
VUE項(xiàng)目自動(dòng)檢測(cè)服務(wù)端是否發(fā)布了新版本
本文主要介紹了VUE項(xiàng)目自動(dòng)檢測(cè)服務(wù)端是否發(fā)布了新版本,通過輪詢方式檢測(cè)Vue項(xiàng)目新版本并提示用戶刷新頁(yè)面,具有一定的參考價(jià)值,感興趣的可以了解一下2025-01-01
vue+vuex+axio從后臺(tái)獲取數(shù)據(jù)存入vuex實(shí)現(xiàn)組件之間共享數(shù)據(jù)
這篇文章主要介紹了vue+vuex+axio從后臺(tái)獲取數(shù)據(jù)存入vuex,組件之間共享數(shù)據(jù),非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-04-04
Vue 實(shí)現(xiàn)前進(jìn)刷新后退不刷新的效果
這篇文章主要介紹了Vue 實(shí)現(xiàn)前進(jìn)刷新后退不刷新的效果,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-06-06
vue在路由中驗(yàn)證token是否存在的簡(jiǎn)單實(shí)現(xiàn)
今天小編就為大家分享一篇vue在路由中驗(yàn)證token是否存在的簡(jiǎn)單實(shí)現(xiàn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-11-11
詳解vue中使用vue-quill-editor富文本小結(jié)(圖片上傳)
這篇文章主要介紹了詳解vue中使用vue-quill-editor富文本小結(jié)(圖片上傳),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04

