Vue實(shí)現(xiàn)滾動(dòng)加載更多效果的示例代碼
引言
在現(xiàn)代Web應(yīng)用中,處理大量數(shù)據(jù)的展示是一個(gè)常見(jiàn)的挑戰(zhàn)。無(wú)論是社交媒體的時(shí)間線、電商平臺(tái)的商品列表,還是新聞網(wǎng)站的文章流,如何高效地加載和渲染大量?jī)?nèi)容,同時(shí)保持良好的用戶體驗(yàn),是開(kāi)發(fā)者需要解決的關(guān)鍵問(wèn)題。傳統(tǒng)的做法是將所有數(shù)據(jù)一次性加載到頁(yè)面上,但這會(huì)導(dǎo)致頁(yè)面加載緩慢、內(nèi)存占用過(guò)高,甚至影響瀏覽器的響應(yīng)速度。為了解決這些問(wèn)題,滾動(dòng)加載更多(Infinite Scroll) 和 虛擬列表(Virtual List) 成為了流行的解決方案。
兩種方案的介紹以及區(qū)別
滾動(dòng)加載更多(Infinite Scroll) :
- 工作原理:滾動(dòng)加載更多通過(guò)分頁(yè)的方式工作。當(dāng)用戶滾動(dòng)到頁(yè)面底部或接近底部時(shí),系統(tǒng)會(huì)觸發(fā)一個(gè)新的請(qǐng)求,獲取下一頁(yè)的數(shù)據(jù),并將其動(dòng)態(tài)添加到現(xiàn)有內(nèi)容的末尾。這種方式允許用戶在不中斷瀏覽的情況下,無(wú)縫加載更多內(nèi)容。
- 優(yōu)點(diǎn):實(shí)現(xiàn)簡(jiǎn)單,用戶體驗(yàn)流暢,減少了初始加載時(shí)間。
- 缺點(diǎn):隨著用戶不斷滾動(dòng),頁(yè)面上的DOM元素會(huì)逐漸增加,可能導(dǎo)致內(nèi)存占用過(guò)高,影響性能。對(duì)于非常大的數(shù)據(jù)集,滾動(dòng)加載更多可能會(huì)導(dǎo)致頁(yè)面變得非常沉重,影響用戶體驗(yàn)。
虛擬列表(Virtual List) :
- 工作原理:虛擬列表的核心思想是只渲染當(dāng)前視口(即用戶可以看到的部分)中的內(nèi)容,而不是一次性將所有數(shù)據(jù)加載到DOM中。當(dāng)用戶滾動(dòng)時(shí),虛擬列表會(huì)動(dòng)態(tài)地更新DOM,確保只有可見(jiàn)區(qū)域的內(nèi)容被渲染。此外,虛擬列表通常會(huì)復(fù)用DOM節(jié)點(diǎn),進(jìn)一步減少DOM操作的開(kāi)銷。
- 優(yōu)點(diǎn):即使數(shù)據(jù)量非常大,虛擬列表也能保持良好的性能,因?yàn)樗讳秩究梢?jiàn)區(qū)域的內(nèi)容。它還非常適合處理海量數(shù)據(jù),避免了內(nèi)存占用過(guò)高的問(wèn)題。
- 缺點(diǎn):實(shí)現(xiàn)相對(duì)復(fù)雜,尤其是需要處理滾動(dòng)事件、DOM復(fù)用、邊界條件等。通常建議使用現(xiàn)成的庫(kù)(如
vue-virtual-scroller、react-window等)來(lái)簡(jiǎn)化開(kāi)發(fā)過(guò)程。
滾動(dòng)加載更多的實(shí)現(xiàn)
使用@scroll
1. 實(shí)現(xiàn)
通過(guò)監(jiān)聽(tīng)滾動(dòng)條的變化來(lái)更新頁(yè)碼,實(shí)現(xiàn)加載更多效果;代碼里面有注釋,我就不多說(shuō)啦。需要注意的是父盒子的高度很重要。
以下是代碼(但是先別拿去用,因?yàn)檫€有重要的優(yōu)化)
<template>
<div class="color-squares" @scroll="handleScroll">
<!-- 這里我使用的是一個(gè)數(shù)字并非是一個(gè)數(shù)組,如果使用的是數(shù)組的話可將size換成arr.slice(0,size)(arr是你的數(shù)組); -->
<div v-for="(item, index) in size" :key="index" class="square">
{{ item }}
</div>
<!-- 末尾方式加載狀態(tài),這里我使用的時(shí)t-design(騰訊組件庫(kù)) -->
<t-loading :loading="loading" text="正在加載更多..." />
</div>
</template>
<script setup>
import { ref, onMounted, computed } from "vue";
//總數(shù)據(jù)量
const total = ref(15);
//我們手動(dòng)模擬分頁(yè)效果,
const pageSize = ref(5);
const pageNum = ref(1);
const loading = ref(false);
//使用計(jì)算屬性來(lái)更改size的值。通過(guò)滾動(dòng)來(lái)更改pageNum(頁(yè)碼)。
const size = computed(() => {
return pageNum.value * pageSize.value > total
? total
: pageNum.value * pageSize.value;
});
//加載更多函數(shù),注意里面的打印信息。
const loadMore = () => {
if (pageNum.value * pageSize.value >= total.value) {
loading.value = false;
return;
}
loading.value = true;
//請(qǐng)注意我
console.log(1);
// 模擬異步請(qǐng)求
setTimeout(() => {
pageNum.value++;
loading.value = false;
}, 1000); // 模擬網(wǎng)絡(luò)延遲
};
const handleScroll = (event) => {
const container = event.target;
if (
container.scrollTop + container.clientHeight >=
container.scrollHeight - 5
) {
loadMore();
}
};
// 生命周期鉤子
onMounted(() => {});
</script>
<style scoped>
.color-squares {
height: 100vh;
overflow-y: scroll;
display: flex;
flex-direction: column;
align-items: center;
text-align: center;
line-height: 100px;
width: 100%;
gap: 10px;
border: 1px solid #ccc;
}
.square {
width: 200px;
height: 200px;
background-color: #ececec;
text-align: center;
line-height: 200px;
border-radius: 8px;
font-size: 24px;
font-weight: bolder;
}
</style>
2. 上述的缺陷以及優(yōu)化
直接看效果圖,我們可以看到我們輕輕滾動(dòng)一下,loadMore就觸發(fā)了多次。注意控制臺(tái)打印信息打印的13個(gè)1,意味著一次滾動(dòng)觸發(fā)了loadMore 13次

所以我們理所當(dāng)然需要給他加上防抖
<script setup>
import { ref, onMounted, computed } from "vue";
//總數(shù)據(jù)量
const total = ref(15);
//我們手動(dòng)模擬分頁(yè)效果,
const pageSize = ref(5);
const pageNum = ref(1);
const loading = ref(false);
//使用計(jì)算屬性來(lái)更改size的值。通過(guò)滾動(dòng)來(lái)更改pageNum(頁(yè)碼)。
const size = computed(() => {
return pageNum.value * pageSize.value > total
? total
: pageNum.value * pageSize.value;
});
//加載更多函數(shù),注意里面的打印信息。
const loadMore = () => {
if (pageNum.value * pageSize.value >= total.value) {
loading.value = false;
return;
}
loading.value = true;
//請(qǐng)注意我
console.log(1);
// 模擬異步請(qǐng)求
setTimeout(() => {
pageNum.value++;
loading.value = false;
}, 1000); // 模擬網(wǎng)絡(luò)延遲
};
function debounce(fn, delay, immediate = false) {
let timer;
return function (...args) {
const context = this;
// 如果設(shè)置了 immediate,并且這是第一次觸發(fā),則立即執(zhí)行目標(biāo)函數(shù)
if (immediate && !timer) {
fn.apply(context, args);
}
// 清除之前的定時(shí)器
if (timer) clearTimeout(timer);
// 設(shè)置新的定時(shí)器
timer = setTimeout(() => {
if (!immediate) {
fn.apply(context, args); // 如果沒(méi)有立即執(zhí)行,則在延遲后執(zhí)行
}
timer = null; // 清空定時(shí)器標(biāo)識(shí)符
}, delay);
};
}
const handleScroll = debounce((event) => {
const container = event.target;
if (
container.scrollTop + container.clientHeight >=
container.scrollHeight - 5
) {
loadMore();
}
}, 300);
// 生命周期鉤子
onMounted(() => {});
</script>
加了防抖的效果

使用 Element Plus 的無(wú)限滾動(dòng)指令 (v-infinite-scroll)(最快)
在需要添加無(wú)限滾動(dòng)的盒子中加上 v-infinite-scroll指令該指令需要綁定一個(gè)函數(shù)。無(wú)需設(shè)置防抖,無(wú)需監(jiān)聽(tīng)滾動(dòng),拿來(lái)即用。
具體代碼如下:
<template>
<div class="color-squares" max-height="500px" v-infinite-scroll="loadMore">
<!-- 如果使用的是數(shù)組,可以將 size 換成 arr.slice(0, size) -->
<div v-for="(item, index) in size" :key="index" class="square">
{{ item }}
</div>
<!-- 末尾方式加載狀態(tài),這里我使用的時(shí) t-design(騰訊組件庫(kù)) -->
<t-loading :loading="loading" text="正在加載更多..." />
</div>
</template>
<script setup>
import { ref, onMounted, computed } from "vue";
//總數(shù)據(jù)量
const total = ref(15);
//我們手動(dòng)模擬分頁(yè)效果,
const pageSize = ref(5);
const pageNum = ref(1);
const loading = ref(false);
//使用計(jì)算屬性來(lái)更改size的值。通過(guò)滾動(dòng)來(lái)更改pageNum(頁(yè)碼)。
const size = computed(() => {
return pageNum.value * pageSize.value > total
? total
: pageNum.value * pageSize.value;
});
//加載更多函數(shù),注意里面的打印信息。
const loadMore = () => {
if (pageNum.value * pageSize.value >= total.value) {
loading.value = false;
return;
}
loading.value = true;
//請(qǐng)注意我
console.log(1);
// 模擬異步請(qǐng)求
setTimeout(() => {
pageNum.value++;
loading.value = false;
}, 1000); // 模擬網(wǎng)絡(luò)延遲
};
</script>
<style scoped>
.color-squares {
display: flex;
flex-direction: column;
align-items: center;
text-align: center;
line-height: 100px;
width: 100%;
gap: 10px;
border: 1px solid #ccc; /* 可選:添加邊框以便更清晰地看到滾動(dòng)容器 */
}
.square {
width: 200px;
height: 200px;
background-color: #ececec;
text-align: center;
line-height: 200px;
border-radius: 8px;
font-size: 24px;
font-weight: bolder;
}
</style>
看看效果圖:

結(jié)語(yǔ)
通過(guò)本期文章,我們掌握了兩種常見(jiàn)的滾動(dòng)加載更多實(shí)現(xiàn)方式:手動(dòng)處理滾動(dòng)事件 和 使用 Element Plus 的無(wú)限滾動(dòng)指令。這兩種方法各有優(yōu)劣,適用于不同的場(chǎng)景。在下一期文章中,我們將進(jìn)一步探討如何使用 Intersection Observer API 來(lái)檢測(cè)元素的重疊度,從而實(shí)現(xiàn)更加靈活和高效的滾動(dòng)加載更多功能。
以上就是Vue實(shí)現(xiàn)滾動(dòng)加載更多效果的示例代碼的詳細(xì)內(nèi)容,更多關(guān)于Vue滾動(dòng)加載更多效果的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
vue對(duì)于低版本瀏覽器兼容問(wèn)題的解決思路
很多時(shí)候使用vue開(kāi)發(fā)的項(xiàng)目,由于無(wú)法在低版本瀏覽器上運(yùn)行,所以需要解決下,下面這篇文章主要給大家介紹了關(guān)于vue對(duì)于低版本瀏覽器兼容問(wèn)題的解決思路,需要的朋友可以參考下2023-02-02
Vue使用高德地圖搭建實(shí)時(shí)公交應(yīng)用功能(地圖 + 附近站點(diǎn)+線路詳情 + 輸入提示+換乘詳情)
這篇文章主要介紹了vue中使用高德地圖搭建實(shí)時(shí)公交應(yīng)用(地圖 + 附近站點(diǎn)+線路詳情 + 輸入提示+換乘詳情),主要是讓大家熟悉下高德地圖在vue中的使用及vue的常用指令,需要的朋友可以參考下2018-05-05
vue3封裝簡(jiǎn)易的vue-echarts問(wèn)題
這篇文章主要介紹了vue3封裝簡(jiǎn)易的vue-echarts問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05
vue ssr+koa2構(gòu)建服務(wù)端渲染的示例代碼
這篇文章主要介紹了vue ssr+koa2構(gòu)建服務(wù)端渲染的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
使用Vue-scroller頁(yè)面input框不能觸發(fā)滑動(dòng)的問(wèn)題及解決方法
這篇文章主要介紹了使用Vue-scroller頁(yè)面input框不能觸發(fā)滑動(dòng)的問(wèn)題,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08

