一、dependentObservable.js:主要包含ko.computed相關方法的處理
二、dependencyDetection.js:主要包含依賴的監控上下文對象。app
一、首先爲某個屬性定義 一個computed對象,以下源碼:
less
var vModel = function(){ this.fName = ko.observable('fName'), this.lName= ko.observable('lName'), this.name= ko.computed(function () { //監控依賴對象 return this.fName() + '-' + this.lName(); },this); };
var state = { latestValue: undefined, isStale: true, isBeingEvaluated: false, suppressDisposalUntilDisposeWhenReturnsFalse: false, isDisposed: false, pure: false, isSleeping: false, readFunction: options["read"], evaluatorFunctionTarget: evaluatorFunctionTarget || options["owner"], disposeWhenNodeIsRemoved: options["disposeWhenNodeIsRemoved"] || options.disposeWhenNodeIsRemoved || null, disposeWhen: options["disposeWhen"] || options.disposeWhen, domNodeDisposalCallback: null, dependencyTracking: {}, dependenciesCount: 0, evaluationTimeoutInstance: null };
// Inherit from 'subscribable' if (!ko.utils.canSetPrototype) { // 'subscribable' won't be on the prototype chain unless we put it there directly ko.utils.extend(computedObservable, ko.subscribable['fn']); } ko.subscribable['fn'].init(computedObservable); //執行發佈/訂閱對象的init方法,用於初始化發佈/訂閱對象。 // Inherit from 'computed' ko.utils.setPrototypeOfOrExtend(computedObservable, computedFn);
var isInitial = state.pure ? undefined : !state.dependenciesCount, // If we're evaluating when there are no previous dependencies, it must be the first time dependencyDetectionContext = { computedObservable: computedObservable, disposalCandidates: state.dependencyTracking, disposalCount: state.dependenciesCount }; ko.dependencyDetection.begin({ callbackTarget: dependencyDetectionContext, callback: computedBeginDependencyDetectionCallback, computed: computedObservable, isInitial: isInitial });
try { var readFunction = state.readFunction; return state.evaluatorFunctionTarget ? readFunction.call(state.evaluatorFunctionTarget) : readFunction(); } finally { ko.dependencyDetection.end(); // For each subscription no longer being used, remove it from the active subscriptions list and dispose it if (dependencyDetectionContext.disposalCount && !state.isSleeping) { ko.utils.objectForEach(dependencyDetectionContext.disposalCandidates, computedDisposeDependencyCallback); } state.isStale = false; }
function observable() { if (arguments.length > 0) { // Write // Ignore writes if the value hasn't changed if (observable.isDifferent(observable[observableLatestValue], arguments[0])) { observable.valueWillMutate(); observable[observableLatestValue] = arguments[0]; observable.valueHasMutated(); } return this; // Permits chained assignments } else { debugger; // Read ko.dependencyDetection.registerDependency(observable); //執行依賴 return observable[observableLatestValue]; } }
registerDependency: function (subscribable) { //注入到相關依賴屬性 if (currentFrame) { if (!ko.isSubscribable(subscribable)) throw new Error("Only subscribable things can act as dependencies"); currentFrame.callback.call(currentFrame.callbackTarget, subscribable, subscribable._id || (subscribable._id = getId())); } }
if (state.disposeWhenNodeIsRemoved && computedObservable.isActive()) { ko.utils.domNodeDisposal.addDisposeCallback(state.disposeWhenNodeIsRemoved, state.domNodeDisposalCallback = function () { computedObservable.dispose(); }); }
一、ko.dependencyDetection中有ignore方法,他主要實現的是一個異步鎖,讓callbcak處於鎖的狀態執行
dom
ignore: function (callback, callbackTarget, callbackArgs) { //按順序s執行依賴,但不觸發訂閱。 try { begin(); return callback.apply(callbackTarget, callbackArgs || []); } finally { end(); } }