vue獲取v-for異步數(shù)據(jù)dom的解決問題
vue獲取v-for異步數(shù)據(jù)dom
問題描述
每次加載界面時(shí),在 mounted
階段,只能獲取普通dom(指靜態(tài)渲染的dom),獲取不到v-for的dom,盡管使用$nextTick
也獲取不到,雖然使用setTimeOut
能解決,但這種方法真的很low,作為程序員要追求完美,下面我們看看怎么完美解決這個(gè)問題。
原因
在v-for里的數(shù)據(jù)是異步獲取的,里面的id或者class也都是動(dòng)態(tài)綁定的,而mounted階段只是實(shí)例掛載完成,這時(shí)候異步請(qǐng)求的數(shù)據(jù)大概率還沒請(qǐng)求完成,從而v-for要渲染的dom肯定也還沒開始渲染,這時(shí)候連異步請(qǐng)求的數(shù)據(jù)都獲取不到,更別說需要異步數(shù)據(jù)來渲染的dom了。
而$nextTick
是指整個(gè)視圖渲染完成,注意這里不包含異步請(qǐng)求的數(shù)據(jù),在整個(gè)視圖渲染完成時(shí)異步數(shù)據(jù)未必能請(qǐng)求到。所以調(diào)用$nextTick
其實(shí)也沒什么兩樣。
解決思路及方案
第一種并不是很完美的方案:使用updated鉤子
updated 階段是完成了數(shù)據(jù)更新到 DOM 的階段(對(duì)加載回來的數(shù)據(jù)進(jìn)行處理),此時(shí),在使用document等獲取dom是可以得到的。
updated 與 mounted 不同的是,在每一次的 DOM 結(jié)構(gòu)更新,Vue.js 都會(huì)調(diào)用一次 updated 鉤子函數(shù)!而 mounted 鉤子函數(shù)僅僅只執(zhí)行一次而已。
但是,本人不推薦這么做,因?yàn)槿绻?xiàng)目中數(shù)據(jù)更新很頻繁,虛擬dom也會(huì)跟著經(jīng)常更新,如果在updated中獲取dom,會(huì)造成操作過于頻繁的問題,也就會(huì)多多少少影響程序性能。在vue官方也有提到。
完美解決的方案:在異步數(shù)據(jù)獲取完成時(shí)調(diào)用$nextTick獲取dom
想獲取異步請(qǐng)求數(shù)據(jù)渲染的dom就一定得等到數(shù)據(jù)接收到后再進(jìn)行相應(yīng)的操作。而有同學(xué)就問了,為什么在獲取到數(shù)據(jù)后還要調(diào)$nextTick
呢?
因?yàn)楂@取到數(shù)據(jù)之后v-for去動(dòng)態(tài)渲染dom肯定也是需要一定時(shí)間的,不可能馬上就能渲染完成,所以在獲取到異步數(shù)據(jù)之后再等整個(gè)視圖渲染完成時(shí)去獲取dom是最好的選擇。
那要在created
或者mounted
或者在watch
中獲取就看個(gè)人選擇了。
在created
或者mounted
中需要在獲取完數(shù)據(jù)后在then
或者await
接受返回的數(shù)據(jù)再調(diào)用$nextTick
獲取dom,在watch中就直接監(jiān)視請(qǐng)求的數(shù)據(jù),數(shù)據(jù)改變就說明獲取到了,就可以調(diào)用$nextTick
獲取dom了。
vue獲取DOM問題
利用 ref 和 $refs 可以用于獲取 dom 元素
<h1 id="h" ref="myH">我是一個(gè)孤獨(dú)可憐又能吃的h1</h1> ? mounted() { ? ? console.log(document.getElementById("h")); ? ? console.log(this.$refs.myH); ? },
使用ref調(diào)用組件內(nèi)的方法。
<Demo ref="de"></Demo> ? import Demo from "./child/demo"; ? //組件起別名ref let demoObj = this.$refs.de; //可以以此調(diào)用組件內(nèi)的方法,變量等 ? ? demoObj.fn();
$nextTick使用
如果我們修改某個(gè)出現(xiàn)在DOM里的變量,之后又要獲取DOM里的變量值,那么會(huì)發(fā)現(xiàn)DOM里變量值并未修改成功,因?yàn)閂ue更新DOM是異步的,用$nextTick解決。
因?yàn)関ue 通過異步隊(duì)列控制 DOM 更新和 nextTick 回調(diào)函數(shù)先后執(zhí)行。
<p ref="myP">{{ count }}</p> <button @click="btn">點(diǎn)擊count+1, 馬上提取p標(biāo)簽內(nèi)容</button> ? btn() { ? ? ? this.count++; // vue監(jiān)測(cè)數(shù)據(jù)更新, 開啟一個(gè)DOM更新隊(duì)列(異步任務(wù)) ? ? ? console.log(this.count) //1 ? ? ? //因?yàn)閂ue更新DOM是異步的,所以count雖然完成加一,但這里的DOM還未更新。 ? ? ? console.log(this.$refs.myP.innerHTML); // 0 ? ? ? ? // 解決: this.$nextTick() ? ? ? // 過程: DOM更新完會(huì)挨個(gè)觸發(fā)$nextTick里的函數(shù)體 ? ? ? this.$nextTick(() => { ? ? ? ? console.log(this.$refs.myP.innerHTML); // 1 ? ? ? }); ? ? ? ? //方法二: async 函數(shù)執(zhí)行時(shí), async btn(){} ? ? ? //如果遇到 await 就會(huì)先暫停執(zhí)行 ,等到觸發(fā)的異步操作完成后, ? ? ? //恢復(fù) async 函數(shù)的執(zhí)行并返回解析值 ? ? ? await this.$nextTick() ? ? ? this.$refs.myP.innerHTML ? },
組件name屬性使用
我們封裝的組件-可以自己定義name屬性組件名-讓使用者有個(gè)統(tǒng)一的前綴風(fēng)格
1.給組件(Demo)起個(gè)名字,用組件的name屬性值, 來注冊(cè)組件名字
export default { ? name: "Mmm", }
2.之后就可以用Mmm當(dāng)作標(biāo)簽使用
import Demo from "./child/demo"; ? components: { ? ? // Demo, ? ? [Demo.name]: Demo, ? },
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
vue?懸浮窗且?guī)ё詣?dòng)吸附功能實(shí)現(xiàn)demo
這篇文章主要為大家介紹了vue?懸浮窗且?guī)ё詣?dòng)吸附功能實(shí)現(xiàn)demo,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06vue 動(dòng)態(tài)創(chuàng)建組件的兩種方法
這篇文章主要介紹了vue 動(dòng)態(tài)創(chuàng)建組件的兩種方法,幫助大家更好的理解和使用vue框架,感興趣的朋友可以了解下2020-12-12Vue實(shí)現(xiàn)下載文件而非瀏覽器直接打開的方法
對(duì)于瀏覽器來說,文本、圖片等可以直接打開的文件,不會(huì)進(jìn)行自動(dòng)下載,下面這篇文章主要給大家介紹了關(guān)于Vue實(shí)現(xiàn)下載文件而非瀏覽器直接打開的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05vue3輸入框生成的時(shí)候如何自動(dòng)獲取焦點(diǎn)詳解
記錄一下自己最近開發(fā)vue3.0的小小問題,下面這篇文章主要給大家介紹了關(guān)于vue3輸入框生成的時(shí)候如何自動(dòng)獲取焦點(diǎn)的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-09-09Vue開發(fā)配置tsconfig.json文件的實(shí)現(xiàn)
tsconfig.json文件中指定了用來編譯這個(gè)項(xiàng)目的根文件和編譯選項(xiàng),本文就來介紹一下Vue開發(fā)配置tsconfig.json文件的實(shí)現(xiàn),感興趣的可以了解一下2023-08-08一步步教你搭建VUE+VScode+elementUI開發(fā)環(huán)境
這篇文章主要給大家介紹了關(guān)于搭建VUE+VScode+elementUI開發(fā)環(huán)境的相關(guān)資料,近期被配置環(huán)境的事情弄得整個(gè)人都要炸了,現(xiàn)在整理如下,希望有相同需求的朋友可以不用走彎路,需要的朋友可以參考下2023-07-07關(guān)于delete和Vue.delete的區(qū)別及說明
這篇文章主要介紹了關(guān)于delete和Vue.delete的區(qū)別及說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10vite+vue3項(xiàng)目解決低版本兼容性問題解決方案(Safari白屏)
這篇文章主要介紹了vite+vue3項(xiàng)目解決低版本兼容性問題(Safari白屏),使用官方插件 @vitejs/plugin-legacy 為打包后的文件提供傳統(tǒng)瀏覽器兼容性支持,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2024-03-03Vue3實(shí)現(xiàn)地圖選點(diǎn)組件的示例代碼
這篇文章主要為大家詳細(xì)介紹了Vue3實(shí)現(xiàn)地圖選點(diǎn)組件的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-01-01