位置: 编程技术 - 正文

JavaScript数据绑定实现一个简单的 MVVM 库(js绑定onchange)

编辑:rootadmin

推荐整理分享JavaScript数据绑定实现一个简单的 MVVM 库(js绑定onchange),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:js三种绑定方式,javascript数据绑定,js绑定onchange,js绑定,js怎么绑定数据,js怎么绑定数据,javascript数据绑定,js怎么绑定数据,内容如对您有帮助,希望把文章链接给更多的朋友!

推荐阅读:

实现非常简单的js双向数据绑定

MVVM 是 Web 前端一种非常流行的开发模式,利用 MVVM 可以使我们的代码更专注于处理业务逻辑而不是去关心 DOM 操作。目前著名的 MVVM 框架有 vue, avalon , react 等,这些框架各有千秋,但是实现的思想大致上是相同的:数据绑定 + 视图刷新。出于好奇和一颗愿意折腾的心,我自己也沿着这个方向写了一个最简单的 MVVM 库 ( mvvm.js ),总共 多行代码,指令的命名和用法与 vue 相似,在这里分享一下实现的原理以及我的代码组织思路。

思路整理

MVVM 在概念上是真正将视图与数据逻辑分离的模式,ViewModel 是整个模式的重点。要实现 ViewModel 就需要将数据模型(Model)和视图(View)关联起来,整个实现思路可以简单的总结成 5 点:

实现一个 Compiler 对元素的每个节点进行指令的扫描和提取;

实现一个 Parser 去解析元素上的指令,能够把指令的意图通过某个刷新函数更新到 dom 上(中间可能需要一个专门负责视图刷新的模块)比如解析节点 <p v-show="isShow"></p> 时先取得 Model 中 isShow 的值,再根据 isShow 更改 node.style.display 从而控制元素的显示和隐藏;

实现一个 Watcher 能将 Parser 中每条指令的刷新函数和对应 Model 的字段联系起来;

实现一个 Observer 使得能够对对象的所有字段进行值的变化监测,一旦发生变化时可以拿到最新的值并触发通知回调;

利用 Observer 在 Watcher 中建立一个对 Model 的监听 ,当 Model 中的一个值发生变化时,监听被触发,Watcher 拿到新值后调用在步骤 2 中关联的那个刷新函数,就可以实现数据变化的同时刷新视图的目的。

效果示例

首先粗看下最终的使用示例,与其他 MVVM 框架的实例化大同小异:

模块划分

我把 MVVM 分成了五个模块去实现: 编译模块 Compiler 、解析模块 Parser 、视图刷新模块 Updater 、数据订阅模块 Watcher 和 数据监听模块 Observer 。流程可以简述为:Compiler 编译好指令后将指令信息交给解析器 Parser 解析,Parser 更新初始值并向 Watcher 订阅数据的变化,Observer 监测到数据的变化然后反馈给 Watcher ,Watcher 再将变化结果通知 Updater 找到对应的刷新函数进行视图的刷新。

上述流程如图所示:

下文就介绍下这五个模块实现的基本原理(代码只贴重点部分,完整的实现请到我的 Github 翻阅)

1. 编译模块 Compiler

Compiler 的职责主要是对元素的每个节点进行指令的扫描和提取。因为编译和解析的过程会多次遍历整个节点树,所以为了提高编译效率在 MVVM 构造函数内部先将 element 转成一个文档碎片形式的副本 fragment 编译对象是这个文档碎片而不应该是目标元素,待全部节点编译完成后再将文档碎片添加回到原来的真实节点中。

vm.complieElement 实现了对元素所有节点的扫描和指令提取:

vm.compileAllNodes 方法将会对 this.$unCompileNodes 中的每个节点进行编译(将指令信息交给 Parser ),编译完一个节点后就从缓存队列中移除它,同时检查 this.$unCompileNodes.length 当 length === 0 时说明全部编译完成,可以将文档碎片追加到真实节点上了。

2. 指令解析模块 Parser

JavaScript数据绑定实现一个简单的 MVVM 库(js绑定onchange)

当编译器 Compiler 把每个节点的指令提取出来后就可以给到解析器解析了。每一个指令都有不同的解析方法,所有指令的解析方法只要做好两件事:一是将数据值更新到视图上(初始状态),二是将刷新函数订阅到 Model 的变化监测中。这里以解析 v-text 为例描述一个指令的大致解析方法:

3. 数据订阅模块 Watcher

上个例子,Watcher 提供了一个 watch 方法来对数据变化进行订阅,一个参数是模型字段 model 另一个是回调函数,回调函数是要通过 Observer 来触发的,参数传入新值 last 和 旧值 old , Watcher 拿到新值后就可以找到 model 对应的回调(刷新函数)进行更新视图了。model 和 刷新函数是一对多的关系,即一个 model 可以有任意多个处理它的回调函数(刷新函数),比如: v-text="title" 和 v-html="title" 两个指令共用一个数据模型字段。

添加数据订阅 watcher.watch 实现方式为:

当数据模型的 field 字段发生改变时,Watcher 就会触发缓存数组中订阅了 field 的所有回调。

4. 数据监听模块 Observer

Observer 是整个 mvvm 实现的核心基础,看过有一篇文章说 O.o (Object.observe) 将会引爆数据绑定革命,给前端带来巨大影响力,不过很可惜,ES7 草案已经将 O.o 给废弃了!目前也没有浏览器支持!所幸的是还有 Object.defineProperty 通过拦截对象属性的存取描述符(get 和 set) 可以模拟一个简单的 Observer :

然后还有个问题就是数组操作 ( push, shift 等) 该如何监测?所有的 MVVM 框架都是通过重写该数组的原型来实现的:

这个实现方式是从 vue 中参考来的,觉得用的很妙,不过数组的 length 属性是不能够被监听到的,所以在 MVVM 中应避免操作 array.length

5. 视图刷新模块 Updater

Updater 在五个模块中是最简单的,只需要负责每个指令对应的刷新函数即可。其他四个模块经过一系列的折腾,把最后的成果交给到 Updater 进行视图或者事件的更新,比如 v-text 的刷新函数为:

v-bind:style 的刷新函数:

双向数据绑定的实现

表单元素的双向数据绑定是 MVVM 的一个最大特点之一:

其实这个神奇的功能实现原理也很简单,要做的只有两件事:一是数据变化的时候更新表单值,二是反过来表单值变化的时候更新数据,这样数据的值就和表单的值绑在了一起。

数据变化更新表单值利用前面说的 Watcher 模块很容易就可以做到:

表单变化更新数据只需要实时监听表单的值得变化事件并更新数据模型对应字段即可:

其他表单 radio, checkbox 和 select 都是一样的原理。

以上,整个流程以及每个模块的基本实现思路都讲完了,第一次在社区发文章,语言表达能力不太好,如有说的不对写的不好的地方,希望大家能够批评指正!

结语

折腾这个简单的 mvvm.js 是因为原来自己的框架项目中用的是 vue.js 但是只是用到了它的指令系统,一大堆功能只用到四分之一左右,就想着只是实现 data-binding 和 view-refresh 就够了,结果没找这样的 javascript 库,所以我自己就造了这么一个轮子。

虽说功能和稳定性远不如 vue 等流行 MVVM 框架,代码实现可能也比较粗糙,但是通过造这个轮子还是增长了很多知识的 ~ 进步在于折腾嘛!

目前我的 mvvm.js 只是实现了最本的功能,以后我会继续完善、健壮它,如有兴趣欢迎一起探讨和改进~

Javascript实现苹果悬浮虚拟按钮 Javascript实现苹果悬浮虚拟按钮直接引入代码到页面即可代码有部分冗余的地方,有兴趣的小伙伴可也自己修改如果有什么BUG记得评论告诉我哦web-touch.jsvar

javascript原生ajax写法分享 ajax:一种请求数据的方式,不需要刷新整个页面;ajax的技术核心是XMLHttpRequest对象;ajax请求过程:创建XMLHttpRequest对象、连接服务器、发送请求、接收

JavaScript实现图片自动加载的瀑布流效果 先给大家展示下效果图:向下滑动网页的时候能够自动加载图片并显示。盛放图片的盒子模型如下:divclass="box"divclass="box_img"imgsrc="Img/8.jpg"/div/div设置img-

标签: js绑定onchange

本文链接地址:https://www.jiuchutong.com/biancheng/373419.html 转载请保留说明!

上一篇:JS表单验证的代码(常用)(js表单验证代码Email)

下一篇:Javascript实现苹果悬浮虚拟按钮(js实现apply函数)

  • 小规模合作社免税吗
  • 非关联企业借款利息扣除
  • 员工为公司垫付款项
  • 租入厂房再转租账要怎么做
  • 红字发票可以只开金额没有数量吗
  • 房地产企业季度所得税申报
  • 销售净水器营业执照范围
  • 调整汇算清缴利息怎么算
  • 小规模纳税人印花税怎样计算
  • 同一控制下企业合并和非同一控制下企业合并的区别
  • 某超市从电器生产厂购进一批电器
  • 准予抵扣的进项税额计算公式
  • 外汇税收怎么缴纳增值税
  • 本月开的发票次月预缴如何做会计分录呢?
  • 职工福利基金提取规定
  • 净现金流量率计算公式
  • 固定资产报废做账
  • 缴纳补充公积金的外企
  • 贷款利息收入要减去支付利息支出吗
  • 社保可以在税前扣除吗
  • 以前年度费用本年开发票
  • 电器以旧换新的套路
  • 信用减值损失编码
  • 电子发票服务平台诺诺发票官网
  • 诉讼费计入哪里
  • 定向增发是什么融资方式
  • 本年利润余额的计算公式
  • win7缓存设置方法
  • win10怎么调出命令行窗口
  • 税金及附加多计提所得税年报怎么更正
  • php登录注册
  • 上市公司发行股票会计分录
  • php实现查询功能
  • 公司取得的发明专利
  • 印花税按次按月
  • 产生转让损失
  • 委托加工物资账务怎么做
  • PHP:mcrypt_module_get_supported_key_sizes()的用法_Mcrypt函数
  • 外贸出口退税进项发票勾选
  • vue框架基础知识
  • 业务招待费用列支范围
  • vue 实战
  • php中的函数
  • php数组函数有哪些
  • vscode怎么看错误提示
  • 深圳杯2020c题
  • 总公司如何成立子公司
  • 印花税应计入哪里
  • 现金折扣什么时候冲减收入
  • 不具备独立核算条件的行政单位
  • mongodb findandmodify
  • 农业技术服务费税率
  • 公司安排异地培训
  • 无形资产账面价值计算公式
  • 收取物业费如何纳税
  • 保证金159001收益如何计算
  • 在外地设办事处需要办什么手续
  • 年底结账会计要怎么做账
  • 无发票的费用怎么算
  • 工会经费属于什么会计科目?
  • 成本暂估入帐如何做分录
  • 净资产增长率增长说明什么
  • 个体户不交税会判刑吗?
  • 新手会计做账怎么做账
  • 外贸整个流程图
  • 加载dll错误是什么意思
  • win10预览版选哪个
  • webcolct.exe - webcolct 是什么进程
  • linux中的rpm什么意思
  • win10 系统账户
  • win7系统电脑开机黑屏
  • 2015-04-04---CCAction详解(欠了大家好几天了)
  • html收藏夹导入
  • python获取entry里输入的值
  • jquery easyui datagrid实现增加,修改,删除方法总结
  • 河南准生证查询平台
  • 企业房产税优惠减免政策2020
  • 贵州地方税务局网上办税服务厅
  • 残疾人在国企有补贴吗
  • 全款房契税退税需要什么资料
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设