原因:防止多個元件實例物件之間共用一個data,產生資料污染;採用函數的形式,initData時會將其作為工廠函數都會傳回全新data物件。當將元件中的data寫成一個函數,資料以函數傳回值形式定義,這樣每復用一次元件,就會傳回一份新的data,擁有自己的作用域,類似於為每個元件實例建立一個私有的資料空間,讓各個元件實例維護各自的資料。
本教學操作環境:windows7系統、vue3版,DELL G3電腦。
一、實例和元件定義data的區別
vue實例的時候定義data屬性既可以是一個對象,也可以是一個函數
const app = new Vue({ el:"#app", // 对象格式 data:{ foo:"foo" }, // 函数格式 data(){ return { foo:"foo" } } })
元件中定義data屬性,只能是一個函數
如果為元件data直接定義為一個物件
Vue.component('component1',{ template:`<div>组件</div>`, data:{ foo:"foo" }})
則會得到警告訊息
警告說明:傳回的data應該是一個函數在每一個元件實例中
二、元件data定義函數與物件的差異
上面講到元件data必須是函數,不知道大家有沒有思考過這是為什麼呢?
在我們定義好一個元件的時候,vue最後都會透過Vue.extend()構成元件實例
這裡我們模仿元件建構函數,定義data屬性,採用物件的形式
function Component(){ } Component.prototype.data = { count : 0 }
建立兩個元件實例
const componentA = new Component() const componentB = new Component()
修改componentA元件data屬性的值,componentB中的值也發生了改變
console.log(componentB.data.count) // 0 componentA.data.count = 1 console.log(componentB.data.count) // 1
產生這樣的原因這是兩者共用了同一個記憶體位址,componentA修改的內容,同樣對componentB產生了影響。 【學習影片分享:vue影片教學、web前端影片】
#如果我們採用函數的形式,則不會出現這種情況(函數傳回的對象記憶體位址並不相同)
function Component(){ this.data = this.data() } Component.prototype.data = function (){ return { count : 0 } }
修改componentA元件data屬性的值,componentB中的值不受影響
console.log(componentB.data.count) // 0 componentA.data.count = 1 console.log(componentB.data.count) // 0
vue元件可能會有很多個實例,採用函數傳回一個全新data形式,使每個實例物件的資料不會受到其他實例物件資料的污染
三、原理分析
首先可以看看vue初始化data的程式碼,data的定義可以是函數也可以是物件
原始碼位置:/vue-dev/src/core/instance/state.js
function initData (vm: Component) { let data = vm.$options.data data = vm._data = typeof data === 'function' ? getData(data, vm) : data || {} ... }
data既能是object也能是function,那為什麼還會出現上文警告呢?
別急,繼續看下文
元件在建立的時候,會進行選項的合併
原始碼位置:/vue-dev/src/core/ util/options.js
自訂元件會進入mergeOptions進行選項合併
Vue.prototype._init = function (options?: Object) { ... // merge options if (options && options._isComponent) { // optimize internal component instantiation // since dynamic options merging is pretty slow, and none of the // internal component options needs special treatment. initInternalComponent(vm, options) } else { vm.$options = mergeOptions( resolveConstructorOptions(vm.constructor), options || {}, vm ) } ... }
定義data會進行資料校驗
原始碼位置:/ vue-dev/src/core/instance/init.js
這時候vm實例為undefined,進入if判斷,若data類型不是function,則出現警告提示
strats.data = function ( parentVal: any, childVal: any, vm?: Component ): ?Function { if (!vm) { if (childVal && typeof childVal !== "function") { process.env.NODE_ENV !== "production" && warn( 'The "data" option should be a function ' + "that returns a per-instance value in component " + "definitions.", vm ); return parentVal; } return mergeDataOrFn(parentVal, childVal); } return mergeDataOrFn(parentVal, childVal, vm); };
四、結論
根實例物件data可以是物件也可以是函數(根實例是單例),不會產生資料污染情況
元件實例物件data必須為函數,目的是為了防止多個元件實例物件之間共用一個data,產生資料污染。採用函數的形式,initData時會將其作為工廠函數都會傳回全新data物件
說明:
- ##vue中元件是用來復用的,為了防止data復用,將其定義為函數。
- vue元件中的data資料都應該是互相隔離,互不影響的,元件每復用一次,data資料就應該被複製一次,之後,當某處複用的地方元件內data資料改變時,其他複用地方元件的data資料不受影響,就需要透過data函數傳回一個物件作為元件的狀態。
- 當我們將元件中的data寫成一個函數,資料以函數傳回值形式定義,這樣每複用一次元件,就會傳回一份新的data,擁有自己的作用域,類似於為每個元件實例建立一個私有的資料空間,讓各個元件實例維護各自的資料。
- 當我們元件的date單純的寫成物件形式,這些實例用的是同一個建構函數,由於JavaScript的特性所導致,所有的元件實例共用了一個data,就會造成一個變了全都會變的結果。
以上是vue元件中data為啥是函數的詳細內容。更多資訊請關注PHP中文網其他相關文章!

前端有没有现成的库,可以直接用来绘制 Flowable 流程图的?下面本篇文章就跟小伙伴们介绍一下这两个可以绘制 Flowable 流程图的前端库。

vue不是前端css框架,而是前端JavaScript框架。Vue是一套用于构建用户界面的渐进式JS框架,是基于MVVM设计模式的前端框架,且专注于View层。Vue.js的优点:1、体积小;2、基于虚拟DOM,有更高的运行效率;3、双向数据绑定,让开发者不用再去操作DOM对象,把更多的精力投入到业务逻辑上;4、生态丰富、学习成本低。

Vue3如何更好地使用qrcodejs生成二维码并添加文字描述?下面本篇文章给大家介绍一下Vue3+qrcodejs生成二维码并添加文字描述,希望对大家有所帮助。

本篇文章我们来了解 Vue2.X 响应式原理,然后我们来实现一个 vue 响应式原理(写的内容简单)实现步骤和注释写的很清晰,大家有兴趣可以耐心观看,希望对大家有所帮助!


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

Safe Exam Browser
Safe Exam Browser是一個安全的瀏覽器環境,安全地進行線上考試。該軟體將任何電腦變成一個安全的工作站。它控制對任何實用工具的訪問,並防止學生使用未經授權的資源。

DVWA
Damn Vulnerable Web App (DVWA) 是一個PHP/MySQL的Web應用程序,非常容易受到攻擊。它的主要目標是成為安全專業人員在合法環境中測試自己的技能和工具的輔助工具,幫助Web開發人員更好地理解保護網路應用程式的過程,並幫助教師/學生在課堂環境中教授/學習Web應用程式安全性。 DVWA的目標是透過簡單直接的介面練習一些最常見的Web漏洞,難度各不相同。請注意,該軟體中

SublimeText3 英文版
推薦:為Win版本,支援程式碼提示!

EditPlus 中文破解版
體積小,語法高亮,不支援程式碼提示功能

SublimeText3 Linux新版
SublimeText3 Linux最新版