preface
The pain of writing this article is comparable to that of writing an essay, which coincides with the end of the novel. I used to think that the author could write thousands of words every day. Now look at yourself, no problem a vulgar code farmer, write a code after reading analysis can not write out.
start
The basic principle, we may have seen more or less, what render time get time collect dependency, set time trigger dependency, rerender to update the view, back can be back out. However, how to collect dependencies, how to trigger dependencies, how to prevent a component from repeating render when updating multiple data data, how to asynchronously render these, may not understand very clearly. Hope that through the following these pale words, can let everyone a little ground, more understanding. (Leaving out special cases (such as readonly, shallow, etc.) does not interfere with the main flow.)
Track and Trigger are good brothers
function track(target, type, key) {
let depsMap = targetMap.get(target);
if(! depsMap) { targetMap.set(target, (depsMap =new Map()));
}
let dep = depsMap.get(key);
if(! dep) { depsMap.set(key, (dep =new Set()));
}
if (!dep.has(activeEffect)) {
dep.add(activeEffect);
activeEffect.deps.push(dep);
}
}
Copy the code
In track, targetMap and depsMap are used to hold the dependencies for each key in an object, respectively. DepsMap => depsMap; depsMap => depsMap; depsMap => depsMap; depsMap => depsMap; depsMap => depsMap; Through track, we can collect the dependency corresponding to each key.
function trigger(target, type, key, newValue, oldValue, oldTarget) {
const depsMap = targetMap.get(target);
if(! depsMap) {// never been tracked
return;
}
const effects = new Set(a);const add = (effectsToAdd) = > {
if (effectsToAdd) {
effectsToAdd.forEach(effect= > {
if(effect ! == activeEffect || effect.allowRecurse) { effects.add(effect); }}); }};else {
// schedule runs for SET | ADD | DELETE
if(key ! = =void 0) { add(depsMap.get(key)); }}const run = (effect) = > {
if (effect.options.scheduler) {
effect.options.scheduler(effect);
}
else{ effect(); }}; effects.forEach(run); }Copy the code
In trigger, we retrieve the effect dependency array we collected in Track by changing the key, and then run these side effect functions in turn.
What effect is a
function effect(fn, options = shared.EMPTY_OBJ) {
const effect = createReactiveEffect(fn, options);
effect();
return effect;
}
Copy the code
function createReactiveEffect(fn, options) {
const effect = function reactiveEffect() {
if(! effectStack.includes(effect)) { cleanup(effect);try {
enableTracking();
effectStack.push(effect);
activeEffect = effect;
return fn();
}
finally {
effectStack.pop();
resetTracking();
activeEffect = effectStack[effectStack.length - 1]; }}}; effect.deps = []; effect.options = options;return effect;
}
Copy the code
This is the truncated effect code, which simply creates a side effect function. Then Effect has its own dependency DEPS, also an array, which is interdependent with the DEPs in track.
So where do we use effect?
const setupRenderEffect = (instance) = > {
// create reactive effect for rendering
instance.update = reactivity.effect(function componentEffect() {
if(! instance.isMounted) {// render mountComponent
}
else {
// render updateComponent
}
}, createDevEffectOptions(instance));
};
Copy the code
We can see that the component’s corresponding update function is actually an effect and then internally differentiates the component state to determine whether to run mount or update logic.
In fact, if you look at this, you can also get a general idea of the principle and. In the effect function, we can see that the first time the component is created, it will run directly because the component has not yet been mounted. Instance. isMounted branch logic. Then collect the dependencies in Render via track. Then, when the data is updated, the effect is found in the trigger by key, which triggers the effect and runs the logic of the update branch of the function above to update the view.
How does Effect avoid repeating runs
When I saw this, I did have this question, because some handlers may change multiple values at one time, but in the DEPS of each value in track, the render logic of the corresponding component is bound. When trigger, this problem is inevitable.
In the createReactiveEffect function, there is a phrase called cleanup(effect), which is triggered while effect is running.
function cleanup(effect) {
const { deps } = effect;
if (deps.length) {
for (let i = 0; i < deps.length; i++) {
deps[i].delete(effect);
}
deps.length = 0; }}Copy the code
The logic is easy to understand. In track, Effect also saves a copy of all dePs that depend on it. As long as effect is cleared in these DEPs, reruning problems can be avoided.
How does Effect run asynchronously
As we all know, VUE updates the DOM asynchronously. First, there must be a task queue that stores all effects and then runs asynchronously.
function createDevEffectOptions(instance) {
return {
scheduler: queueJob,
allowRecurse: true.onTrack: instance.rtc ? e= > shared.invokeArrayFns(instance.rtc, e) : void 0.onTrigger: instance.rtg ? e= > shared.invokeArrayFns(instance.rtg, e) : void 0
};
}
Copy the code
In trigger, we see that there is an effect.options.scheduler(effect), which is passed to the component when it creates effect.
function queueJob(job) {
if((! queue.length || ! queue.includes(job, isFlushing && job.allowRecurse ? flushIndex +1: flushIndex)) && job ! == currentPreFlushParentJob) {const pos = findInsertionIndex(job);
if (pos > -1) {
queue.splice(pos, 0, job);
}
else{ queue.push(job); } queueFlush(); }}function queueFlush() {
if(! isFlushing && ! isFlushPending) { isFlushPending =true; currentFlushPromise = resolvedPromise.then(flushJobs); }}Copy the code
The principle is simple: each effect is added to the queue and then run asynchronously through promise.then. (flushJobs is the exact logic that runs a queue.)
conclusion
The writing is not very detailed, and many details are omitted in the middle. I may analyze the details in more detail in the future. I think I am thankful for my ability to write here. After all, reading the source code is a very painful thing, in the source code roam, it is easy to be overwhelmed, drowned.
If there is any error, sorry in advance, because are looking at their own source analysis. Once you’re done Posting, give yourself a thumbs-up on your little site and it’s LV2, something to celebrate. When the original flag was LV2, I left ningbo and went to Hangzhou. I thought it would drag on until the end of the year, but it was done in four months. Still make me this vegetable dog feel gratified. To become from 0574 mixed 0571, but a qualified MC will not forget where he came from ~
Finally, I hope I can get the desired offer, I hope every coarse code farmers have a good career prospects ~