在日常開發中,
$set
的也是一個非常實用的API,因為Vue2實作響應式的核心是利用了ES5的Object.defineProperty
,當我們透過直接修改數組下標更改數組或為物件添加新的屬性,這時候Object.defineproperty是監聽不到資料的變化的,這時候大家就會用上$set
,讓修改的操作也實現響應,我們知其然更要知其所以然,接下來看一下Vue中的$set是如何實現的。 【相關推薦:vuejs影片教學、web前端開發】
#應用程式場景
let dataArr = ["item1"]; let dataObject = { name: "ccs" }; dataArr[2] = "item2"; dataObject.age = 22; 响应失败,页面没有显示更新新增的数据 this.$set(this.dataArr,2,'item2') this.$set(this.dataObject,'age',22) 响应成功,页面显示更新新增的数据
set實作
接下來我們來看看$set在Vue中的定義
function set(target: Array<any> | Object, key: any, val: any): any { if ( process.env.NODE_ENV !== "production" && (isUndef(target) || isPrimitive(target)) ) { warn( `Cannot set reactive property on undefined, null, or primitive value: ${(target: any)}` ); } if (Array.isArray(target) && isValidArrayIndex(key)) { target.length = Math.max(target.length, key); target.splice(key, 1, val); return val; } if (key in target && !(key in Object.prototype)) { target[key] = val; return val; } const ob = (target: any).__ob__; if (target._isVue || (ob && ob.vmCount)) { process.env.NODE_ENV !== "production" && warn( "Avoid adding reactive properties to a Vue instance or its root $data " + "at runtime - declare it upfront in the data option." ); return val; } if (!ob) { target[key] = val; return val; } defineReactive(ob.value, key, val); ob.dep.notify(); return val; }
在原始碼中先判斷set的目標是否是undefined
和基本型別
如果是 undefined
或基本類型
就報錯,
因為使用者不應該往undefined和基本類型中set東西,
然後又判斷了目標是否是數組與key是不是合法的index,合法的index是指值為大於等於0的整數,
如果兩個條件都成立就對目標數組調用splice方法插入或者修改數組
,
這裡的splice
不是普通的splice
,是王維詩裡的splice,是被vue代理人重寫過的splice
#陣列實作回應
$set實作陣列修改回應的方式是代理人重寫的陣列的一部分方法,接下來我們看一下具體實作
const arrayProto = Array.prototype export const arrayMethods = Object.create(arrayProto) const methodsToPatch = [ 'push', 'pop', 'shift', 'unshift', 'splice', 'sort', 'reverse' ] function def(obj, key, val, enumerable) { Object.defineProperty(obj, key, { value: val, enumerable: !!enumerable, writable: true, configurable: true }); } methodsToPatch.forEach(function (method) { const original = arrayProto[method] def(arrayMethods, method, function mutator (...args) { const result = original.apply(this, args) const ob = this.__ob__ let inserted switch (method) { case 'push': case 'unshift': inserted = args break case 'splice': inserted = args.slice(2) break } if (inserted) ob.observeArray(inserted) ob.dep.notify() return result }) })
vue中代理重寫的不只是splice
,有push、pop、shift、unshift、splice、sort、reverse
這七個方法,
首先執行了const result = original.apply(this, args)
#執行原本數組的方法並取得它的值,接下來判斷如果是往數組中新增值就將新加入的值也實作響應式
,
最後一步拿到這個陣列的_ob_物件
對_ob_
裡的dep進行派發更新。
想深入了解vue的回應式可以查閱往期文章
面試官問你Vue2的回應式原理,你怎麼答? - 掘金(juejin.cn)
物件實作回應
$set
中下半部的邏輯就是用來處理物件回應的,我們接著往下看
if (key in target && !(key in Object.prototype)) { target[key] = val; return val; } const ob = (target: any).__ob__; if (!ob) { target[key] = val; return val; } defineReactive(ob.value, key, val); ob.dep.notify(); return val;
首先判斷了屬性如果在目標物件中直接return結束邏輯,
因為vue只有在新增目標物件中原本沒有的屬性時才會失去回應,例如let obj={} obj.name='ccs'
,
vue在初始化的時候會將data裡的所有屬性都變成響應式,如果的值是物件或陣列則會new一個Observer
實例儲存在__ob__,想深入了解vue的回應式可以查閱往期文章
面試官問你Vue2的響應式原理,你怎麼答? - 掘金(juejin.cn)
拿到這個物件的_ob_進行判斷,如果不存在就表示是未經過vue初始化的普通物件而不是回應式物件否則就手動透過defineReactive
為屬性新增get方法與set方法實作回應,
然後手動呼叫dep
裡的notify()
發布更新。
總結
vue中$set方法對數組和物件的處理本質上的一樣的,對新增的值添加回應然後手動觸發派發更新。
以上是聊聊Vue中$set是如何實現的?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

whenthevue.jsvirtualdomdetectschange,itupdatesthevirlualdom,diffsit和appliesminimalchangeStothereAldom.thisprocessensuresrocessensureshighhighpperformance byformance byavoidingunnnnnnnnnnneclastory dommaniplastions。

Vue.js的VirtualDOM既是真實DOM的鏡像,又不完全是。 1.創建和更新:Vue.js基於組件定義創建VirtualDOM樹,狀態變化時先更新VirtualDOM。 2.差異和修補:通過diff操作比較新舊VirtualDOM,僅將最小變化應用到真實DOM。 3.效率:VirtualDOM允許批量更新,減少直接DOM操作,優化渲染過程。 VirtualDOM是Vue.js優化UI更新的戰略工具。

Vue.js和React在可擴展性和可維護性上的表現各有優勢。 1)Vue.js易於上手,適合小型項目,CompositionAPI提升了大型項目可維護性。 2)React適用於大型複雜項目,Hooks和虛擬DOM提高了性能和可維護性,但學習曲線較陡峭。

Vue.js和React的未來趨勢和預測分別是:1)Vue.js將在企業級應用中廣泛應用,並在服務端渲染和靜態站點生成方面有突破;2)React將在服務器組件和數據獲取方面創新,並進一步優化並發模式。

Netflix的前端技術棧主要基於React和Redux。 1.React用於構建高性能的單頁面應用,通過組件化開發提升代碼重用性和維護性。 2.Redux用於狀態管理,確保狀態變化可預測和可追踪。 3.工具鏈包括Webpack、Babel、Jest和Enzyme,確保代碼質量和性能。 4.性能優化通過代碼分割、懶加載和服務端渲染實現,提升用戶體驗。

Vue.js是一種漸進式框架,適用於構建交互性強的用戶界面。其核心功能包括響應式系統、組件化開發和路由管理。 1)響應式系統通過Object.defineProperty或Proxy實現數據監聽,自動更新界面。 2)組件化開發允許將界面拆分為可複用的模塊。 3)VueRouter支持單頁面應用,提升用戶體驗。

Vue.js的主要缺點包括:1.生態系統相對較新,第三方庫和工具不如其他框架豐富;2.學習曲線在復雜功能上變得陡峭;3.社區支持與資源不如React和Angular廣泛;4.大型應用中可能遇到性能問題;5.版本升級與兼容性挑戰較大。

Netflix使用React作為其前端框架。 1.React的組件化開發和虛擬DOM機制提高了性能和開發效率。 2.使用Webpack和Babel優化代碼構建和部署。 3.採用代碼分割、服務端渲染和緩存策略進行性能優化。


熱AI工具

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

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

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

Dreamweaver Mac版
視覺化網頁開發工具

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

ZendStudio 13.5.1 Mac
強大的PHP整合開發環境

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具