vant-list組件觸發(fā)多次onload事件導(dǎo)致數(shù)據(jù)亂序的解決方案
1、問題描述
3個(gè)tab頁簽切換時(shí),調(diào)用不同接口,在某一個(gè)tab只要翻頁到>=2的情況,當(dāng)它再點(diǎn)擊到另一個(gè)tab的時(shí)候,另外一個(gè)tab就會(huì)連續(xù)調(diào)用兩次查詢接口,有可能第二次查詢的數(shù)據(jù)先于第一次查詢返回,這時(shí)兩次數(shù)據(jù)concat后順序就不對(duì)了(第二次查到的數(shù)據(jù)在前,第一次在后)
原因分析
因?yàn)樗粌H觸發(fā)了created還觸發(fā)了onload事件:且順序?yàn)椋哼M(jìn)入created調(diào)用查詢接口,在.then之前,異步查詢還未返回的時(shí)候,又會(huì)去觸發(fā)onload事件,在current+=1之后再次進(jìn)行查詢
2、先了解一下相關(guān)知識(shí)點(diǎn)
1)基礎(chǔ)用法
List 組件通過 loading 和 finished 兩個(gè)變量控制加載狀態(tài),當(dāng)組件滾動(dòng)到底部時(shí),會(huì)觸發(fā) load事件并將 loading 設(shè)置成 true。
此時(shí)可以發(fā)起異步操作并更新數(shù)據(jù),數(shù)據(jù)更新完畢后,將 loading 設(shè)置成 false 即可。
若數(shù)據(jù)已全部加載完畢,則直接將 finished 設(shè)置成 true 即可。
2)List 的運(yùn)行機(jī)制是什么?
List 會(huì)監(jiān)聽瀏覽器的滾動(dòng)事件并計(jì)算列表的位置,當(dāng)列表底部與可視區(qū)域的距離小于offset時(shí),List會(huì)觸發(fā)一次 load 事件。
3)為什么 List 初始化后會(huì)立即觸發(fā) load 事件?
List 初始化后會(huì)觸發(fā)一次 load事件,用于加載第一屏的數(shù)據(jù),這個(gè)特性可以通過immediate-check屬性關(guān)閉。
4)為什么會(huì)連續(xù)觸發(fā) load 事件?
如果一次請(qǐng)求加載的數(shù)據(jù)條數(shù)較少,導(dǎo)致列表內(nèi)容無法鋪滿當(dāng)前屏幕,List 會(huì)繼續(xù)觸發(fā) load事件,直到內(nèi)容鋪滿屏幕或數(shù)據(jù)全部加載完成。
因此你需要調(diào)整每次獲取的數(shù)據(jù)條數(shù),理想情況下每次請(qǐng)求獲取的數(shù)據(jù)條數(shù)應(yīng)能夠填滿一屏高度。
5)loading 和 finished 分別是什么含義?
List有以下三種狀態(tài),理解這些狀態(tài)有助于你正確地使用List組件:
- 非加載中,loading為false,此時(shí)會(huì)根據(jù)列表滾動(dòng)位置判斷是否觸發(fā)onload事件(列表內(nèi)容不足一屏幕時(shí),會(huì)直接觸發(fā))
- 加載中,loading為true,表示正在發(fā)送異步請(qǐng)求,此時(shí)不會(huì)觸發(fā)onload事件
- 加載完成,finished為true,此時(shí)不會(huì)觸發(fā)onload事件 在每次請(qǐng)求完畢后,需要手動(dòng)將loading設(shè)置為false,表示加載結(jié)束
這次遇到的問題,可以利用第五點(diǎn),解決方法:
在進(jìn)入created的發(fā)送請(qǐng)求之前,將this.loading = true , ⇒⇒⇒ (利用:加載中,loading為true,表示正在發(fā)送異步請(qǐng)求,此時(shí)不會(huì)觸發(fā)load事件),讓系統(tǒng)知道此時(shí)正在異步請(qǐng)求數(shù)據(jù),讓它別觸發(fā)onload事件
3、代碼
<van-list v-model="loading" class="van-list-style" :immediate-check="false" :finished="finished" :finished-text="finishedText" :error.sync="error" error-text="請(qǐng)求失敗,點(diǎn)擊重新加載" @load="onLoad"> <div class="list" v-if="dataList.length > 0"> <div class="list-box2" v-for="(item,index) in dataList" :key="index" @click="handleClick(item)"> ……………………………………… </div> </div> <div v-if="noData" style="margin-top:30%"> <img src="@/common/imgs/no-data.png" alt="" style="width:100%"> </div> </van-list> data() { return { dataList:[], current:1, size:10, loading: false, // 上拉加載 ?????? finished: false, // 上拉加載完畢 error: false, // 是否展示錯(cuò)誤 finishedText:"沒有更多了", noData:false, // 是否展示沒有數(shù)據(jù)的圖片 // offset: 100 // 滾動(dòng)條與底部距離小于 offset 時(shí)觸發(fā)load事件 } }, created() { // 調(diào)用列表查詢接口 this.init(this.tab) }, methods:{ init(val) { if(val === '1') { this.createList(1,10) } }, //查詢接口 createList(current,size) { let param = { current:current, size:size } // 重點(diǎn)!?。。。。。。?!在這里將loading置為true this.loading = true; createList(param).then(res=>{ let that = this if(res.status == true) { //賦值 const dataList = res.body.records const pages = res.body.pages // 如果返回?cái)?shù)據(jù)為空 if(dataList == null || dataList.length === 0) { that.finished = true that.finishedText = "沒發(fā)現(xiàn)任何東西,去其他地方逛逛吧~" that.noData = true return } // 加載狀態(tài)結(jié)束 可以寫在這里也可以寫在finally里面 that.loading = false; // 根據(jù)當(dāng)前頁進(jìn)行數(shù)據(jù)處理 if(that.current === 1) { that.dataList = dataList } else { that.dataList = that.dataList.concat(dataList) } // xxx?。?!最后一頁不足10條的情況 ,這樣寫實(shí)際有問題,因?yàn)槿绻亲詈笠豁摓?0條的情況,就會(huì)第二次去調(diào)用接口 //if(dataList.length < that.size) { // that.finished = true //that.finishedText = '沒有更多了'; //} // 使用這種??! 證明已經(jīng)是最后一頁了 if(that.current === pages) { that.finished = true that.finishedText = '沒有更多了'; } } }) .catch(err=>{this.error = true; }) .finally(()=>{this.loading = false }) }, // 上拉刷新 onLoad() { this.current+=1 this.createList(this.current,this.size) }, }
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
vue項(xiàng)目打包優(yōu)化方式(讓打包的js文件變小)
這篇文章主要介紹了vue項(xiàng)目打包優(yōu)化方式(讓打包的js文件變小),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-04-04vue3.0使用taro-ui-vue3引入組件不生效的問題及解決
這篇文章主要介紹了vue3.0使用taro-ui-vue3引入組件不生效的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03vue+openlayer5獲取當(dāng)前鼠標(biāo)滑過的坐標(biāo)實(shí)現(xiàn)方法
在vue項(xiàng)目中怎么獲取當(dāng)前鼠標(biāo)劃過的坐標(biāo)呢?下面通過本文給大家分享實(shí)現(xiàn)步驟,感興趣的朋友跟隨小編一起看看吧2021-11-11Vue項(xiàng)目如何根據(jù)不同運(yùn)行環(huán)境打包項(xiàng)目
這篇文章主要介紹了Vue項(xiàng)目如何根據(jù)不同運(yùn)行環(huán)境打包項(xiàng)目問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03修改vue源碼實(shí)現(xiàn)動(dòng)態(tài)路由緩存的方法
這篇文章主要介紹了修改vue源碼實(shí)現(xiàn)動(dòng)態(tài)路由緩存的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01