今天就跟大家聊聊有關(guān)Vue.js中computed的工作原理是什么,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
創(chuàng)新互聯(lián)公司是一家集網(wǎng)站建設(shè),濂溪企業(yè)網(wǎng)站建設(shè),濂溪品牌網(wǎng)站建設(shè),網(wǎng)站定制,濂溪網(wǎng)站建設(shè)報價,網(wǎng)絡(luò)營銷,網(wǎng)絡(luò)優(yōu)化,濂溪網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競爭力??沙浞譂M足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時我們時刻保持專業(yè)、時尚、前沿,時刻以成就客戶成長自我,堅持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實用型網(wǎng)站。
JS屬性:
JavaScript有一個特性是 Object.defineProperty
,它能做很多事,但我在這篇文章只專注于這個方法中的一個:
var person = {}; Object.defineProperty (person, 'age', { get: function () { console.log ("Getting the age"); return 25; } }); console.log ("The age is ", person.age); // Prints: // // Getting the age // The age is 25
(Obeject.defineProperty是Object的一個方法,第一個參數(shù)是對象名稱,第二個參數(shù)是要設(shè)置的屬性名,第三個參數(shù)是一個對象,它可以設(shè)置這個屬性是否可修改、可寫等,而這篇文章主要使用的是Obeject.defineProperty的訪問器屬性,感興趣的朋友可以自行g(shù)oogle或者查看Js高及程序設(shè)計)
盡管 person.age 看起來像是訪問了對象的一個屬性,但其實在內(nèi)部我們是運(yùn)行了一個函數(shù)。
一個基本可響應(yīng)的Vue.js
Vue.js內(nèi)部構(gòu)建了一個可以將普通的對象轉(zhuǎn)化為可以被觀察的值( 響應(yīng)屬性 ),下面為大家展示一個簡化版的如何添加響應(yīng)屬性的案例:
function defineReactive (obj, key, val) { Object.defineProperty (obj, key, { get: function () { return val; }, set: function (newValue) { val = newValue; } }) }; // 創(chuàng)建一個對象 var person = {}; // 添加可響應(yīng)的屬性"age"和"country" defineReactive (person, 'age', 25); defineReactive (person, 'country', 'Brazil'); // 現(xiàn)在你可以隨意使用person.age了 if (person.age < 18) { return 'minor'; } else { return 'adult'; } // 設(shè)置person.country的值 person.country = 'Russia';
有趣的是, 25 和 ‘Brazil' 還是一個閉包內(nèi)部的變量,只有當(dāng)賦給它們新值的時候 val 才會改變。 person.country 并不擁有 'Brazil' 這個值,而是getter這個函數(shù)擁有 'Brazil' 這個值。
聲明一個計算屬性
讓我們創(chuàng)建一個定義計算屬性的函數(shù) defineComputed 。這個函數(shù)就跟大家平時使用computed時的一樣。
defineComputed ( person, // 計算屬性就聲明在這個對象上 'status', // 計算屬性的名稱 function () { // 實際返回計算屬性值的函數(shù) console.log ("status getter called") if (person.age < 18) { return 'minor'; } else { return 'adult'; } }, function (newValue) { // 當(dāng)計算屬性值更新時調(diào)用的函數(shù) console.log ("status has changed to", newValue) } }); // 我們可以像使用一般的屬性一樣使用計算屬性 console.log ("The person's status is: ", person.status);
讓我們寫一個簡單的 defineComputed 函數(shù),它支持調(diào)用計算方法,但目前不需要它支持 updateCallback 。
function defineComputed (obj, key, computeFunc, updateCallback) { Object.defineProperty (obj, key, { get: function () { // 執(zhí)行計算函數(shù)并且返回值 return computeFunc (); }, set: function () { // 什么也不做,不需要設(shè)定計算屬性的值 } }) }
這個函數(shù)有兩個問題:
每次訪問計算屬性時都會執(zhí)行一次計算函數(shù) computeFunc ()
它不知道什么時候更新 (即當(dāng)我們更新某個data中的屬性,計算屬性中也會更新這個data屬性)
// 我希望最終函數(shù)執(zhí)行后是這個效果:每當(dāng)person.age更新值的時候,person.status也同步更新 person.age = 17; // console: status 的值為 minor person.age = 22; // console: status 的值為 adult
增加一個依賴項
讓我們增加一個全局變量 Dep :
var Dep = { target: null };
這是一個依賴項,接著我們用一個騷操作來更新 defineComputed 函數(shù):
function defineComputed (obj, key, computeFunc, updateCallback) { var onDependencyUpdated = function () { // TODO } Object.defineProperty (obj, key, { get: function () { // 將onDependencyUpdated 這個函數(shù)傳給Dep.target Dep.target = onDependencyUpdated; var value = computeFunc (); Dep.target = null; }, set: function () { // 什么也不做,不需要設(shè)定計算屬性的值 } }) }
現(xiàn)在讓我們回到之前設(shè)置的響應(yīng)屬性上:
function defineReactive (obj, key, val) { // 所有的計算屬性都依賴這個數(shù)組 var deps = []; Object.defineProperty (obj, key, { get: function () { // 檢查是否調(diào)用了計算屬性,如果調(diào)用了,Department.target將等于一個onDependencyUpdated函數(shù) if (Dep.target) { // 把onDependencyUpdated函數(shù)push到deos中 deps.push (target); } return val; }, set: function (newValue) { val = newValue; // 通知所有的計算屬性,告訴它們有個響應(yīng)屬性更新了 deps.forEach ((changeFunction) => { changeFunction (); }); } }) };
我們可以在計算屬性定義的函數(shù)觸發(fā)更新回調(diào)后更新 onDependencyUpdated 函數(shù)。
var onDependencyUpdated = function () { // 再次計算 計算屬性的值 var value = computeFunc (); updateCallback (value); }
把它們整合到一起:
讓我們重新訪問我們的計算屬性 person.status :
person.age = 22; defineComputed ( person, 'status', function () { if (person.age > 18) { return 'adult'; } }, function (newValue) { console.log ("status has changed to", newValue) } }); console.log ("Status is ", person.status);
看完上述內(nèi)容,你們對Vue.js中computed的工作原理是什么有進(jìn)一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝大家的支持。