阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

带你进入Vue2响应式系统之嵌套之旅

285次阅读
没有评论

共计 3970 个字符,预计需要花费 10 分钟才能阅读完成。

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

带你进入 Vue2 响应式系统之嵌套之旅

场景

在 Vue 开发中肯定存在组件嵌套组件的情况,类似于下边的样子。
带你进入 Vue2 响应式系统之嵌套之旅
回到我们之前的响应式系统,模拟一下上边的情况:

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 对象。

带你进入 Vue2 响应式系统之嵌套之旅

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

带你进入 Vue2 响应式系统之嵌套之旅

执行过程
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() {
      /************ 修改的地方 *******************************/
        pushTarget(this); // 保存包装了当前正在执行的函数的 Watcher
       /*******************************************/
        let value;
        try {value = this.getter.call();
        } catch (e) {throw e;} finally {
          /************ 修改的地方 *******************************/
            popTarget();
          /*******************************************/
            this.cleanupDeps();}
        return value;
    }
   ...
}
测试

回到开头的场景,再来执行一下:

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); 的时候将 Watcher 入栈。

带你进入 Vue2 响应式系统之嵌套之旅

进入 updateParentComponent 函数,执行 new Watcher(updateMyComponent); 的时候将 Watcher 入栈。

带你进入 Vue2 响应式系统之嵌套之旅

执行 updateMyComponent 函数,data.inner 收集当前 Dep.target,执行完毕后 Watcher 出栈。

带你进入 Vue2 响应式系统之嵌套之旅

继续执行 updateParentComponent 函数,data.text 收集当前 Dep.target。

此时依赖就变得正常了,data.text 会触发 updateParentComponent 函数,从而输出如下:

 子组件收到: 内部
父组件收到:hello, world
子组件收到: 内部
父组件收到:hello, liang
总结

今天这个相对好理解一些,通过栈解决了嵌套调用的情况。

阿里云 2 核 2G 服务器 3M 带宽 61 元 1 年,有高配

腾讯云新客低至 82 元 / 年,老客户 99 元 / 年

代金券:在阿里云专用满减优惠券

正文完
星哥玩云-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2024-07-25发表,共计3970字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中

星哥玩云

星哥玩云
星哥玩云
分享互联网知识
用户数
4
文章数
19348
评论数
4
阅读量
7802839
文章搜索
热门文章
开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南 大家好,我是星哥。之前介绍了腾讯云的 Code...
星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛 NAS-6:抖音视频同步工具,视频下载自动下载保存 前言 各位玩 NAS 的朋友好,我是星哥!...
云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

云服务器部署服务器面板 1Panel:小白轻松构建 Web 服务与面板加固指南 哈喽,我是星哥,经常有人问我不...
我把用了20年的360安全卫士卸载了

我把用了20年的360安全卫士卸载了

我把用了 20 年的 360 安全卫士卸载了 是的,正如标题你看到的。 原因 偷摸安装自家的软件 莫名其妙安装...
星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

星哥带你玩飞牛 NAS-3:安装飞牛 NAS 后的很有必要的操作 前言 如果你已经有了飞牛 NAS 系统,之前...
阿里云CDN
阿里云CDN-提高用户访问的响应速度和成功率
随机文章
终于收到了以女儿为原型打印的3D玩偶了

终于收到了以女儿为原型打印的3D玩偶了

终于收到了以女儿为原型打印的 3D 玩偶了 前些日子参加某网站活动,获得一次实物 3D 打印的机会,于是从众多...
开源神器组合!1Panel面板+Halo助你轻松打造个人/企业内容中心

开源神器组合!1Panel面板+Halo助你轻松打造个人/企业内容中心

开源神器组合!1Panel 面板 +Halo 助你轻松打造个人 / 企业内容中心 前言 大家好,我是星哥,之前...
CSDN,你是老太太喝粥——无齿下流!

CSDN,你是老太太喝粥——无齿下流!

CSDN,你是老太太喝粥——无齿下流! 大家好,我是星哥,今天才思枯竭,不写技术文章了!来吐槽一下 CSDN。...
安装并使用谷歌AI编程工具Antigravity(亲测有效)

安装并使用谷歌AI编程工具Antigravity(亲测有效)

  安装并使用谷歌 AI 编程工具 Antigravity(亲测有效) 引言 Antigravity...
星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛 NAS-6:抖音视频同步工具,视频下载自动下载保存 前言 各位玩 NAS 的朋友好,我是星哥!...

免费图片视频管理工具让灵感库告别混乱

一言一句话
-「
手气不错
开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

  开源 MoneyPrinterTurbo 利用 AI 大模型,一键生成高清短视频! 在短视频内容...
240 元左右!五盘位 NAS主机,7 代U硬解4K稳如狗,拓展性碾压同价位

240 元左右!五盘位 NAS主机,7 代U硬解4K稳如狗,拓展性碾压同价位

  240 元左右!五盘位 NAS 主机,7 代 U 硬解 4K 稳如狗,拓展性碾压同价位 在 NA...
Prometheus:监控系统的部署与指标收集

Prometheus:监控系统的部署与指标收集

Prometheus:监控系统的部署与指标收集 在云原生体系中,Prometheus 已成为最主流的监控与报警...
如何免费使用强大的Nano Banana Pro?附赠邪修的用法

如何免费使用强大的Nano Banana Pro?附赠邪修的用法

如何免费使用强大的 Nano Banana Pro?附赠邪修的用法 前言 大家好,我是星哥,今天来介绍谷歌的 ...
支付宝、淘宝、闲鱼又双叕崩了,Cloudflare也瘫了连监控都挂,根因藏在哪?

支付宝、淘宝、闲鱼又双叕崩了,Cloudflare也瘫了连监控都挂,根因藏在哪?

支付宝、淘宝、闲鱼又双叕崩了,Cloudflare 也瘫了连监控都挂,根因藏在哪? 最近两天的互联网堪称“故障...