詳解vue.js之props傳遞參數(shù)
本篇文章通過demo實例給大家詳細分析了props傳遞參數(shù)的用法以及遇到問題后的解決辦法,以下是全部內(nèi)容。
前段時間用vue做一個后臺管理系統(tǒng),其中每一頁都需要一個表格來展示信息。自然就想到了將表格提取出來做成公共組件,將不同頁面的數(shù)據(jù)傳入進行渲染,達到復用的目的。
demo地址
1. 問題發(fā)現(xiàn)
在父組件中,需要向表格組件傳遞的數(shù)據(jù)有表格的內(nèi)容數(shù)據(jù)tableData,表格的頁面數(shù)據(jù)page。
<div> <my-table :table-data="tableData" :page-info="pageInfo" id="myTable"></my-table> </div>
其中tableData是個Array對象,是所有需要在表格中展示的數(shù)據(jù)對象組成的一個數(shù)組。而pageInfo是個Object對象,包含了表格頁面信息。在父組件對兩個數(shù)據(jù)對如下初始化,形式如下
tableData:[], pageInfo: { current: 1, // 當前是第幾頁 total: 100, // 數(shù)據(jù)對象的總數(shù) size: 20 // 每頁顯示的數(shù)量 }
按照官方文檔上的說明,prop是單向綁定的,不應該在子組件內(nèi)部改變prop。之所以有想修改prop中數(shù)據(jù)的沖動,主要是prop作為初始值傳入后,子組件想把它當作局部數(shù)據(jù)來用。對于這種情況,官方的說法是定義一個局部變量,并用 prop 的值初始化它:
props: ['tableData', 'pageInfo'], data() { return { tData: this.tableData, page: this.pageInfo } }
然后根據(jù)官方文檔的說法,當每次父組件更新時,子組件的所有prop都會更新為最新值。而tableData和pageInfo的信息是異步通過api從server端獲取的:
{ error: 0, msg: "調(diào)用成功.", data: { restrictioninfo: [...], total: 42 } }
因此當獲取到數(shù)據(jù)時父組件需要改變傳入子組件中的值:
me.tableData = Json.data.restrictioninfo; me.pageInfo.total = Json.data.total;
按理說這時候子組件中的值應該更新成server返回的值,但是子組件頁面的總數(shù)更新了,但table數(shù)據(jù)依然是初始化時的空數(shù)組。(黑人問號???)
。
2.賦值與綁定
首先需要定位數(shù)據(jù)是在哪個地方出了問題,于是我做了一個demo來定位問題。
首先看父組件與子組件中各元素的初始值:
然后當只改變父組件中數(shù)組的引用時可以看到子組件的props數(shù)組隨之改變,而子組件中綁定的數(shù)組確并沒有隨之改變
因此,可以發(fā)現(xiàn),問題是出在了這一步
props: ['tableData', 'pageInfo'], data() { return { tData: this.tableData, page: this.pageInfo } }
而要弄清楚問題的根源,就得弄清楚vue文檔中深入響應式原理。
"在Vue實例的data選項中,Vue將遍歷此對象所有的屬性,并使用Object.defineProperty把這些屬性全部轉(zhuǎn)為 getter/setter","每個組件實例都有相應的 watcher 實例對象,它會在組件渲染的過程中把屬性記錄為依賴,之后當依賴項的 setter 被調(diào)用時,會通知 watcher 重新計算,從而致使它關聯(lián)的組件得以更新。"文檔中說了這么一大堆,簡單理解就是Vue將data選項中的vm.$data.a與DOM中的vm.a進行了雙向綁定,即其中一個變化,另一個也會跟著變化。在Vue源碼中是由defineReactive$$1函數(shù)實現(xiàn)的:
但在子其中主要是利用了Object.defineProperty的get和set方法實現(xiàn)了雙向綁定。而在子組件中,pros數(shù)據(jù)和子組件的$data是通過如下方式聯(lián)系在一起的:
tData: this.tableData
查詢Vue源碼可知this.tableData與tData之間僅僅是賦值,即"="關系
而上述的initData函數(shù)是在組件構(gòu)建時候執(zhí)行的,因此只會在create時執(zhí)行一次。這也是為什么官方文檔中"作為初始值傳入"這一說法,因為他本就只會執(zhí)行一次。當組件構(gòu)建完成后,this.tableData與tData就沒有半毛錢關系了,其中一個的變化并不會引起另一個變化。當然,這種說法并不準確,因為在上文中,我們動態(tài)改變父組件傳入的total,子組件也"隨之"改變,感覺就像是綁定在一起了啊,這又是怎么回事呢?
3.引用類型帶來的假象
當然,我們還是要從官方文檔出發(fā)來解決這個問題。文檔中有這樣一個提示:
這里就需要理解引用類型的概念,引用數(shù)據(jù)類型值指保存在堆內(nèi)存中的對象。也就是,變量中保存的實際上的只是一個指針,這個指針指向內(nèi)存中的另一個位置,該位置保存著對象。訪問方式是按引用訪問。例如一個js對象a,他在內(nèi)存中的存儲形式如下圖所示:
var a = new Object();
當操作時,需要先從棧中讀取內(nèi)存地址,然后再延指針找到保存在堆內(nèi)存中的值再操作。
a.name = 'xz';
引用類型變量賦值,本質(zhì)上賦值的是存儲在棧中的指針,將指針復制到棧中未新變量分配的空間中,而這個指針副本和原指針指向存儲在堆中的同一個對象;賦值操作結(jié)束后,兩個變量實際上將引用同一個對象。因此,在使用時,改變其中的一個變量的值,將影響另一個變量。
var b = a;
在了解了引用類型之后,我們在來看看上文提到的動態(tài)改變傳入子組件前后內(nèi)存中的情況:
me.tableData = Json.data.restrictioninfo; me.pageInfo.total = Json.data.total; ======================================== props: ['tableData', 'pageInfo'], data() { return { tData: this.tableData, page: this.pageInfo } }
首先對tableData的改變是改變了其引用的指針,而對pageInfo則改變了其中一個屬性的值,因此動態(tài)改變前:
動態(tài)改變后:
這樣就解釋了為什么子組件頁面的總數(shù)更新了,但table數(shù)據(jù)依然是初始化時的空數(shù)組。因為引用類型的存在,我們動態(tài)改變父組件傳入的total,子組件也"隨之"改變了。
相關文章
Pinia進階setup函數(shù)式寫法封裝到企業(yè)項目
這篇文章主要為大家介紹了Pinia進階setup函數(shù)式寫法封裝到企業(yè)項目實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-07-07vue項目中字符串換行顯示方式(返回的數(shù)據(jù)包含‘\r\n’字符)
這篇文章主要介紹了vue項目中字符串換行顯示方式(返回的數(shù)據(jù)包含‘\r\n’字符),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-04-04vue中使用 pinia 全局狀態(tài)管理的實現(xiàn)
本文主要介紹了vue中使用 pinia 全局狀態(tài)管理的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-07-07