您现在的位置是:亿华云 > 系统运维

Vue2剥丝抽茧-响应式系统之嵌套

亿华云2025-10-03 11:39:27【系统运维】3人已围观

简介场景在 Vue 开发中肯定存在组件嵌套组件的情况,类似于下边的样子。{{ text}}<div>{{ text}}</div>回到我们之前的响应式系统,模拟一下上边的情况:

场景

在 Vue 开发中肯定存在组件嵌套组件的丝抽情况,类似于下边的茧响样子。

{ { text }}

<div>{ { text }}</div>

回到我们之前的应式响应式系统,模拟一下上边的系统情况:

import { observe } from "./reactive";

import Watcher from "./watcher";

const data = {

text: "hello, world",

inner: "内部",

};

observe(data);

const updateMyComponent = () => {

console.log("子组件收到:", data.inner);

};

const updateParentComponent = () => {

new Watcher(updateMyComponent);

console.log("父组件收到:", data.text);

};

new Watcher(updateParentComponent);

data.text = "hello, liang";

可以先 1 分钟考虑一下上边输出什么?

首先回忆一下 new Watcher 会做什么操作。

第一步是丝抽保存当前函数,然后执行当前函数前将全局的茧响 Dep.target 赋值为当前 Watcher 对象。

接下来执行 getter 函数的应式时候,如果读取了相应的系统属性就会触发 get ,从而将当前 Watcher 收集到该属性的丝抽 Dep 中。

执行过程import { observe } from "./reactive";

import Watcher from "./watcher";

const data = {

text: "hello,茧响 world",

inner: "内部",

};

observe(data);

const updateMyComponent = () => {

console.log("子组件收到:", data.inner);

};

const updateParentComponent = () => {

new Watcher(updateMyComponent);

console.log("父组件收到:", data.text);

};

new Watcher(updateParentComponent);

data.text = "hello, liang";

我们再一步一步理清一下:

new Watcher(updateParentComponent);

将 Dep.target 赋值为保存了 updateParentComponent 函数的 Watcher 。

接下来执行 updateParentComponent 函数。应式

new Watcher(updateMyComponent);

将 Dep.target 赋值为保存了 updateMyComponent 函数的系统 Watcher 。

接下来执行 updateMyComponent 函数。丝抽

const updateMyComponent = () => {

console.log("子组件收到:",茧响 data.inner);

};

// 读取了 inner 变量。服务器托管

// data.inner 的应式 Dep 收集当前 Watcher(保存了 `updateMyComponent` 函数)const updateParentComponent = () => {

new Watcher(updateMyComponent);

console.log("父组件收到:", data.text);

};

// 读取了 text 变量。

// data.text 的 Dep 收集当前 Watcher (保存了 `updateMyComponent` 函数)data.text = "hello, liang";

触发 text 的 set 函数,执行它依赖的 Watcher ,而此时是 updateMyComponent 函数。

所以上边代码最终输出的结果是:

子组件收到: 内部 // new Watcher(updateMyComponent); 时候输出

父组件收到:hello, world // new Watcher(updateParentComponent); 时候输出

子组件收到: 内部 // data.text = "hello, liang"; 输出

然而子组件并不依赖 data.text,依赖 data.text 的父组件反而没有执行。

修复

上边的问题出在我们保存当前正在执行 Watcher 时候使用的是单个变量 Dep.target = null; // 静态变量,全局唯一。

回忆一下学习 C 语言或者汇编语言的时候对函数参数的处理:

function b(p) {

console.log(p);

}

function a(p) {

b("child");

console.log(p);

}

a("parent");

当函数发生嵌套调用的时候,执行 a 函数的源码下载时候我们会先将参数压入栈中,然后执行 b 函数,同样将参数压入栈中,b 函数执行完毕就将参数出栈。此时回到 a 函数就能正确取到 p 参数的值了。

对应于 Watcher 的收集,我们同样可以使用一个栈来保存,执行函数前将 Watcher 压入栈,执行函数完毕后将 Watcher 弹出栈即可。其中,Dep.target 始终指向栈顶 Watcher ,代表当前正在执行的函数。

回到 Dep 代码中,我们提供一个压栈和出栈的方法。

import { remove } from "./util";

let uid = 0;

export default class Dep {

... 省略

}

Dep.target = null; // 静态变量,全局唯一

// The current target watcher being evaluated.

// This is globally unique because only one watcher

// can be evaluated at a time.

const targetStack = [];

export function pushTarget(target) {

targetStack.push(target);

Dep.target = target;

}

export function popTarget() {

targetStack.pop();

Dep.target = targetStack[targetStack.length - 1]; // 赋值为栈顶元素

}

然后 Watcher 中,执行函数之前进行入栈,执行后进行出栈。

import { pushTarget, popTarget } from "./dep";

export default class Watcher {

constructor(Fn) {

this.getter = Fn;

this.depIds = new Set(); // 拥有 has 函数可以判断是否存在某个 id

this.deps = [];

this.newDeps = []; // 记录新一次的站群服务器依赖

this.newDepIds = new Set();

this.get();

}

/

**

* Evaluate the getter, and re-collect dependencies.

*/

get() {

/

很赞哦!(18)