肥宅钓鱼网
当前位置: 首页 钓鱼百科

面试官问你的数据分析(说说你对双向绑定的理解)

时间:2023-07-04 作者: 小编 阅读量: 1 栏目名: 钓鱼百科

说说你对双向绑定的理解本文为面试官VUE系列总进度:3/33一、什么是双向绑定我们先从单向绑定切入单向绑定非常简单,就是把Model绑定到View,当我们用JavaScript代码更新Model时,View就会自动更新双向。

本文为面试官VUE系列总进度:3/33

一、什么是双向绑定

我们先从单向绑定切入

单向绑定非常简单,就是把Model绑定到View,当我们用JavaScript代码更新Model时,View就会自动更新

双向绑定就很容易联想到了,在单向绑定的基础,用户更新了ViewModel的数据也自动被更新了,这种情况就是双向绑定

举个栗子

当用户填写表单时,View的状态就被更新了,如果此时可以自动更新Model的状态,那就相当于我们把ModelView做了双向绑定

关系图如下

二、双向绑定的原理是什么

我们都知道 Vue 是数据双向绑定的框架,双向绑定由三个重要部分构成

  • 数据层(Model):应用的数据及业务逻辑
  • 视图层(View):应用的展示效果,各类UI组件
  • 业务逻辑层(ViewModel):框架封装的核心,它负责将数据与视图关联起来

而上面的这个分层的架构方案,可以用一个专业术语进行称呼:MVVM

这里的控制层的核心功能便是 “数据双向绑定” 。自然,我们只需弄懂它是什么,便可以进一步了解数据绑定的原理

理解ViewModel

它的主要职责就是:

  • 数据变化后更新视图
  • 视图变化后更新数据

当然,它还有两个主要部分组成

  • 监听器(Observer):对所有数据的属性进行监听
  • 解析器(Compiler):对每个元素节点的指令进行扫描跟解析,根据指令模板替换数据,以及绑定相应的更新函数

三、如何实现双向数据

我们还是以Vue为例,先来看看Vue中的双向绑定流程是什么的

  1. new Vue()首先执行初始化,对data执行响应化处理,这个过程发生Observe

  2. 同时对模板执行编译,找到其中动态绑定的数据,从data中获取并初始化视图,这个过程发生在Compile

  3. 同时定义⼀个更新函数和Watcher,将来对应数据变化时Watcher会调用更新函数

  4. 由于data的某个key在⼀个视图中可能出现多次,所以每个key都需要⼀个管家dep来管理多个Watcher

  5. 将来data中数据⼀旦发生变化,会首先找到对应的Dep,通知所有Watcher执行更新函数

流程图如下:

实现

先来一个构造函数:执行初始化,对data执行响应化处理

class Vue {constructor(options) {this.$options = options;this.$data = options.data;// 对data选项做响应式处理observe(this.$data);// 代理data到vm上proxy(this);// 执行编译new Compile(options.el, this);}}

data选项执行响应化具体操作

function observe(obj) {if (typeof obj !== "object" || obj == null) {return;}new Observer(obj);}class Observer {constructor(value) {this.value = value;this.walk(value);}walk(obj) {Object.keys(obj).forEach((key) => {defineReactive(obj, key, obj[key]);});}}

编译Compile

对每个元素节点的指令进行扫描跟解析,根据指令模板替换数据,以及绑定相应的更新函数

class Compile {constructor(el, vm) {this.$vm = vm;this.$el = document.querySelector(el);// 获取domif (this.$el) {this.compile(this.$el);}}compile(el) {const childNodes = el.childNodes;Array.from(childNodes).forEach((node) => { // 遍历子元素if (this.isElement(node)) {// 判断是否为节点console.log("编译元素"node.nodeName);} else if (this.isInterpolation(node)) {console.log("编译插值⽂本"node.textContent);// 判断是否为插值文本 {{}}}if (node.childNodes && node.childNodes.length > 0) {// 判断是否有子元素this.compile(node);// 对子元素进行递归遍历}});}isElement(node) {return node.nodeType == 1;}isInterpolation(node) {return node.nodeType == 3 && /\{\{(.*)\}\}/.test(node.textContent);}}

依赖收集

视图中会用到data中某key,这称为依赖。同⼀个key可能出现多次,每次都需要收集出来用⼀个Watcher来维护它们,此过程称为依赖收集

多个Watcher需要⼀个Dep来管理,需要更新时由Dep统⼀通知

实现思路

  1. defineReactive时为每⼀个key创建⼀个Dep实例
  2. 初始化视图时读取某个key,例如name1,创建⼀个watcher1
  3. 由于触发name1getter方法,便将watcher1添加到name1对应的Dep中
  4. name1更新,setter触发时,便可通过对应Dep通知其管理所有Watcher更新

// 负责更新视图class Watcher {constructor(vm, key, updater) {this.vm = vmthis.key = keythis.updaterFn = updater// 创建实例时,把当前实例指定到Dep.target静态属性上Dep.target = this// 读一下key,触发getvm[key]// 置空Dep.target = null}// 未来执行dom更新函数,由dep调用的update() {this.updaterFn.call(this.vm, this.vm[this.key])}}

声明Dep

class Dep {constructor() {this.deps = [];// 依赖管理}addDep(dep) {this.deps.push(dep);}notify() {this.deps.forEach((dep) => dep.update());}}

创建watcher时触发getter

class Watcher {constructor(vm, key, updateFn) {Dep.target = this;this.vm[this.key];Dep.target = null;}}

依赖收集,创建Dep实例

function defineReactive(obj, key, val) {this.observe(val);const dep = new Dep();Object.defineProperty(obj, key, {get() {Dep.target && dep.addDep(Dep.target);// Dep.target也就是Watcher实例return val;},set(newVal) {if (newVal === val) return;dep.notify(); // 通知dep执行更新方法},});}

参考文献

  • https://www.liaoxuefeng.com/wiki/1022910821149312/1109527162256416
  • https://juejin.cn/post/6844903942254510087#heading-9
  • https://vue3js.cn

    @JS语音答题社群

    推荐阅读
  • 别克汽车上的各种按键功能(别克车型十六种复位功能)

    上海别克音响断电后解锁功能按以下步骤输入密码,在任何两个步骤之间停顿不要超过,现在小编就来说说关于别克汽车上的各种按键功能?重复本程序6、将点火开关置于OFF位置六、赛欧车点火钥匙编程1、将需编程的(全新的)钥匙收发器插入点火开关。表示气压灯已复位。

  • 玉米炒火腿肠的做法(玉米炒火腿肠的做法介绍)

    将火腿肠洗净切好备用;将玉米剥成粒洗净备用;,下面我们就来说一说关于玉米炒火腿肠的做法?我们一起去了解并探讨一下这个问题吧!玉米炒火腿肠的做法将火腿肠洗净切好备用;将玉米剥成粒洗净备用;将青红辣椒洗净切成沫;在锅中加入适量油,将青红辣椒放入锅中;翻炒一会;加入适量食盐;加入切好的火腿肠;加入备好的玉米;炒熟后起锅,美味的玉米炒火腿肠就做好了。

  • 怎样把链接小视频转换成小视频(看这里)

    下面内容希望能帮助到你,我们来一起看看吧!怎样把链接小视频转换成小视频先打开浏览器,找到需要下载的视频。然后复制该视频的地址,并且打开“维棠”网页视频下载软件。在“维棠”软件中找到“下载”选项,并打开。将视频的地址粘贴到“下载连接”方框里。然后点击右下角的“立即下载”即可。

  • 十大最高的金矿(金矿品位最高超9000克)

    已探明斯旺矿段沿倾向460米长,垂向厚度300米。今年以来,柯克兰湖黄金公司勘探投资超过2000万美元,目前现场有10台钻机,其中有两台在斯旺矿段。公司计划在明年一季度更新资源储量。因此该矿的储量和产量均居世界第一位,是名符其实的世界最大金矿。兰德盆地南北长400公里。由于本次钻探持续见矿,该公司已经计划将本次1万米的钻探计划增加至1.26万米。第一阶段钻探结果证实了金梅萨的

  • 大葱如何种植方法(大葱怎样种植方法)

    大葱如何种植方法选择要种植的葱的种类,找到种子,选择一块可以得到全天日照排水良好的地方,挖土到30厘米深,加入堆肥等或其他有机物质。种植床要充分浇水,等幼苗开始出芽时决定是否要间苗,以便留给更多的空间。成熟后的葱间距应在5-7.5厘米,幼苗之间放上护根,用草屑或者树皮等覆盖在幼苗周围的土壤上。如果在盆里种葱,可以跳过这一步,在整个生长季,每周给葱浇约2.5厘米深的水,3-4周之后,葱会长到15-20厘米,这时就可以吃了。

  • 九日齐山登高表达作者什么情感(月明星稀乌鹊南飞)

    曹操是三国时期伟大的政治家、军事家、文学家和书法家。用第一种乌鹊的解释与《短歌行》整体意思毫无关联。曹操在这首诗歌中抒发了自己的政治抱負,表达了自己求贤若渴,爱才惜才的愿望。当时正是曹操带领大軍,与东吴隔江相望,曹操兵精粮足,一切准备就绪。这令曹操喜出望外。俘虏陈琳后,曹操爱其才,委以重任。陈琳为曹操所写的文书,曹操甚至都不增减一字。这就是三国之中独曹操能胜出的根本原因。

  • 失眠睡不着的10个方法(送你3个补救方法)

    人体脚部有很多“管”失眠的穴位,中药泡脚的方式有助于改善睡眠。今天和大家介绍一个调理失眠的方子——安梦足浴包,出自中医著作《自我调养巧治病》。杜仲归肝、肾经。平时要尽可能避免摄入一些辛辣刺激性的食物,还应该避免饮用咖啡、浓茶等多种刺激性的饮品。日常注意多补充一些富含维生素和矿物质营养的食物,睡前两个小时最好不要进食,也不要大量饮水。

  • 蠼螋跑到床上什么原因(蠼螋怎么跑到床上)

    蠼螋跑到床上什么原因?以下内容希望对你有帮助!蠼螋跑到床上什么原因蠼螋跑到床上的原因是床板可能出现腐烂,床的位置偏潮湿和阴暗,蠼螋喜欢待在这种的环境下,它也被叫做剪刀虫,是一种杂食性的昆虫,经常取食植物的花、叶以及枯朽的腐木,也会吃腐败的动植物残体和捕食小昆虫。它没有毒性,对人体无害。家中一般出现蠼螋的地方大多是卧室、卫生间和厨房这几个地方,是因为地面潮湿和不干净而导致的。

  • 引流推广的方法和技巧(推广引流方法有哪些)

    注意保持发布内容的垂直度,做好数据分析,调整优化方案。淘宝店铺的优势:经营成本低,经营方式比较灵活,不限制地区,不需要房租,只需一台电脑就可以做到不间断地营业,面向的消费者范围广。

  • 利亚纳和利亚纳a6区别(利亚纳和利亚纳a6区别简述)

    利亚纳和利亚纳a6区别底盘相同的两款车,利亚纳和利亚纳a6区别主要是外壳,仪表,轮胎,刹车做了优化,原来a+的样子其实是全球统一的样式,a6的外观只有中国有。作为一款实用型家轿,利亚纳a+最大的变化在于搭载了铃木旗下技术成熟的K14B-GVVT发动机,排量更低,油耗更少,但是动力却依然强劲。利亚纳a+工况油耗为5.6L/百公里,在油价居高不下的今天,可谓是一个非常实惠的选择。