【手写 Vue2.x 源码】第二十三篇 - 依赖收集 - 视图更新部分
创始人
2024-05-11 20:53:05
0

一,前言

上篇,主要介绍了依赖收集过程中 dep 和 watcher 关联:

利用 js 单线程特性,在 Watcher 类中 get 方法即将触发视图更新前,
利用全局的类静态属性 Dep.target 记录 Watcher 实例
并且,在视图渲染的取值过程中,在Object.defineProperty的get方法中,让数据的dep记住渲染watcher
从而,实现了 dep 与 watcher 相关联,只有参与视图渲染的数据发生变化才会触发视图的更新

本篇,继续依赖收集的视图更新部分


二,实现视图更新逻辑

1.查重 watcher

问题:同一数据在视图中多次使用会怎样?

按照当前逻辑,同一数据在一个视图中被多次使用时,相同 watcher 会在 dep 中多次保存

  • {{name}}
  • {{name}}
  • {{name}}
  • 这时 name 的 dep 中,会保存三个相同的渲染 watcher

    所以,watcher 需要进行查重

    同 Dep 类的做法:给 Watcher 添加一个 id,
    每次 new Watcher 是 id 自增,以此作为标记对 watcher 实例进行查重
    
    let id = 0;
    class Watcher {constructor(vm, fn, cb, options){this.vm = vm;this.fn = fn;this.cb = cb;this.options = options;this.id = id++;   // watcher 唯一标记this.getter = fn;this.get();}get(){Dep.target = this;this.getter();Dep.target = null;}
    }
    export default Watcher
    

    2. 让 watcher 也记住 dep

    前面,让数据的dep记住了渲染 watcher
    同样的,watcher 也有必要记住 dep

    let id = 0;class Dep {constructor(){this.id = id++;this.subs = [];}// 让 watcher 记住 dep(查重),再让 dep 记住 watcherdepend(){// 相当于 watcher.addDep:使当前 watcher 记住 depDep.target.addDep(this);  }// 让 dep 记住 watcher - 在 watcher 中被调用addSub(watcher){this.subs.push(watcher);}
    }Dep.target = null;  // 静态属性,用于记录当前 watcherexport default Dep;
    

    为什么要这样实现?

    如果要互相记住,watcher 中要对 dep 做查重;dep 中也要对 watcher 做查重;
    用这种方法使 dep 和 watcher 关联在一起后,只要判断一次就可以了
    
    import Dep from "./dep";
    let id = 0;
    class Watcher {constructor(vm, fn, cb, options){this.vm = vm;this.fn = fn;this.cb = cb;this.options = options;this.id = id++;this.depsId = new Set();  // 用于当前 watcher 保存 dep 实例的唯一idthis.deps = []; // 用于当前 watcher 保存 dep 实例this.getter = fn;this.get();}addDep(dep){let did = dep.id;// dep 查重 if(!this.depsId.has(did)){// 让 watcher 记住 depthis.depsId.add(did);this.deps.push(dep);// 让 dep 也记住 watcherdep.addSub(this); }}get(){Dep.target = this;this.getter();Dep.target = null;}
    }export default Watcher;
    

    这种实现方式,会让 dep 和 watcher 保持一种相对的关系:

    如果 watcher 中存过 dep;那么 dep 中一定存过 watcher ;
    如果 watcher 中没存过 dep;那么 dep 中一定没存过 watcher ;
    所以只需要判断一次,就能完成 dep 和 watcher 的查重;
    

    3. 数据改变,触发视图更新

    当视图更新时,会进入 Object.defineProperty 的 set 方法
    需要在 set 方法中,通知 dep 中所有收集的 watcher,执行视图更新方法
    
    // src/observe/index.js#defineReactivefunction defineReactive(obj, key, value) {observe(value);let dep = new Dep();  // 为每个属性添加一个 depObject.defineProperty(obj, key, {get() {if(Dep.target){dep.depend();}return value;},set(newValue) {if (newValue === value) returnobserve(newValue);value = newValue;dep.notify(); // 通知当前 dep 中收集的所有 watcher 依次执行视图更新}})
    }
    

    4. Dep 中添加 notify 方法:

    let id = 0;class Dep {constructor(){this.id = id++;this.subs = [];}depend(){Dep.target.addDep(this);  }addSub(watcher){this.subs.push(watcher);}// dep 中收集的全部 watcher 依次执行更新方法 updatenotify(){this.subs.forEach(watcher => watcher.update())}
    }
    Dep.target = null;
    export default Dep;
    

    5. Watcher 中添加 update 方法:

    import Dep from "./dep";
    let id = 0;
    class Watcher {constructor(vm, fn, cb, options){this.vm = vm;this.fn = fn;this.cb = cb;this.options = options;this.id = id++;this.depsId = new Set();this.deps = [];this.getter = fn;this.get();}addDep(dep){let did = dep.id;if(!this.depsId.has(did)){this.depsId.add(did);this.deps.push(dep);dep.addSub(this); }}get(){Dep.target = this;this.getter();Dep.target = null;}// 执行视图渲染逻辑update(){this.get();}
    }
    export default Watcher;
    

    6. 问题

    多次频繁更新同一个数据,会使视图频繁进行重新渲染操作

    let vm = new Vue({el: '#app',data() {return { name: "Brave" , age: 123}}
    }); 
    vm.name = "Brave Wang";
    vm.name = "Brave";
    vm.name = "Brave Wang";
    vm.name = "Brave";
    vm.name = "Brave Wang";
    vm.name = "Brave";
    
    name的值变化了 6 次,但最终其实没有变化还是 Brave,
    这里就需要改为做异步更新的机制
    

    三,结尾

    本篇,介绍了 Vue依赖收集的视图更新部分,主要涉及以下几点:

    视图初始化时:

    • render方法中会进行取值操作,进入 Object.defineProperty 的 get 方法
    • get 方法中为数据添加 dep,并记录当前的渲染 watcher
    • 记录方式:watcher查重并记住 dep,dep 再记住 watcher

    数据更新时:

    • 当数据发生改,会进入 Object.defineProperty 的 set 方法
    • 在 set 方法中,使 dep 中收集的全部 watcher 执行视图渲染操作 watcher.get()
    • 在视图渲染前(this.getter方法执行前),通过 dep.target 记录当前的渲染 watcher
    • 重复视图初始化流程

    下一篇:Vue 异步更新


    维护日志:

    20210802:修改文章摘要

    相关内容

    热门资讯

    孙杨参赛4项 涵盖短中长距离 转自:天津日报  本报讯(记者 李蓓)昨天,即将在本周末开赛的2025全国游泳冠军赛报项名单出炉。3...
    齐鲁品牌文化数智平台共建项目启...   张志恒 王鑫 济南报道  5月13日举行的“新质山东 品筑未来”2025年山东最具影响力品牌暨《...
    云湖兰山服务综合体运营 转自:贵州日报 本报讯 5月13日,观山湖区环百花湖旅居推介会暨云湖兰山启幕式在观山湖区朱昌...
    守好健康证的“健康关” 人都没到场,证就办好了?近日,有记者走访发现,一些医疗机构的健康证体检项目“缺斤少两”,规章制度形同...
    “我的论文不是AI写的”(图) 转自:天津日报  有网友反映,自己原创的论文经过系统检测之后,竟被指出AI生成内容比例过高。据媒体报...
    应用机器人与3D打印术行颌骨截...   刘通 通讯员 崔子昂 李鲲济南报道  近日,山东大学齐鲁医院口腔科颌面外科陈安威、王涛、韩亦冰及...
    美股周二收盘点评:通货膨胀率下... 来源:宏观对冲陈凯丰Kevin截至4月份的12个月里,消费者物价指数上涨了2.3%,而截至3月份的1...
    纽约汇市:美元下跌 美国通胀率...   彭博一项衡量美元强弱的指数兑所有G10货币均走低,此前发布的美国4月份消费者价格涨幅低于预期。投...
    要为丘陵山区等研制急需急用的装... □四川日报全媒体记者 阚莹莹  近日,四川省农业农村厅发布2025年“天府良机”薄弱环节关键技术装备...
    亮出低空经济发展的未来 □四川日报全媒体记者 高杲 李欣忆7款“四川造”低空产品小鹰-700飞机  ●是目前唯一国产的上单翼...
    聆听“沧海龙吟” 共赏民歌之美... 转自:天津日报  本报讯(记者 张帆 摄影 曹彤)昨天,由首都图书馆发起并联合天津图书馆、河北省图书...
    彩桥即将合龙   5月12日,宜宾市屏山县岷江二桥工地施工繁忙,建设者对桥梁最后几榀钢拱架进行吊装作业。岷江二桥是...
    第二十届西博会5月25日开幕 ●拟邀请匈牙利、老挝为主宾国,浙江、青海为主宾省●拟特邀阿联酋担任大会合作伙伴●据初步统计,西部各地...
    各美其美 美美与共 □四川日报全媒体记者 吴晓铃  在古老的历史长河中,留下帕特农神庙、雅典卫城等文明遗产的希腊是西方文...
    “公证日记” 转自:天津日报  智慧公证  曾经,办理继承公证需要集齐一沓证明材料,群众要跑好几个部门;如今,滨海...
    武清区扎实推进兴业富农 绘就乡... 转自:天津日报  初夏时节,灿烂的阳光夹杂着阵阵微风,深情抚慰着大运河畔的武清区南蔡村镇丁家瞿阝村。...
    如皋农商银行:双向奔赴谋发展 ... 4月27日,如皋农商银行举办“跨境人民币赋能外贸企业发展”专题培训会。会议邀请跨境人民币业务优质企业...
    泰州农商银行联合海陵区供销总社... 日前,泰州农商银行与泰州市海陵区供销总社在城中街道联合举办“政银携手进社区 惠民助农促消费”系列活动...
    将虾苗卖到“小龙虾之乡” □四川日报全媒体记者 陈丽霏  5月,鲜活肥美的小龙虾大量上市,中江县黄鹿镇也迎来了一年中最繁忙的时...
    建圈强链 四川农业明确产业“路... 5月11日,四川省大邑县现代农业(粮食产业)园区,工人在试验田中开展小麦新品的测产工作。 李旭 摄(...