最易理解的VUE双向绑定原理不足70行代码搞定,逐行注释!

VUE双向绑定原理是前端小伙伴很难绕过的一道面试题!本篇文章对其原理进行了最大程度的精简,希望对面试VUE开发的前端小伙伴有所帮助!我在这里将指令 v-改为z-,主要完成z-model、z-click、z-text以及z-html四个提令。

为了能够快速读懂代码,首先要先弄明白以下三个概念:

1、观察者(observer):也就是数据监听器,负责数据对象的所有属性进行监听劫持,并将消息发送给订阅者进行数据更新
2、订阅者(watcher):负责接收数据的变化,更新视图(view),数据与订阅者是一对多的关系。
3、解析器(compile):负责对你的每个节点元素指令进行扫描和解析,负责相关指令的数据初始化及创造订阅者

实现效果如下:


html:
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="lib/zhang.js"></script>
</head>
<body>
    <div id="myApp">
        <input type="button" value="加个!" z-on:click="fn">
        <input type="text" style="width:400px" z-model="site">
        <div z-text="site"></div>
        <div z-html="site"></div>
    </div>
</body>
<script>
    var vm = new Zhang({
        el: "#myApp",
        data: {
            site: "<a href='http://www.zhangpeiyue.com'>zhangpeiyue</a>"
        },
        methods: {
            fn() {
                this.site += "!";
            }
        }
    })
</script>
</html>
zhang.js完整代码如下,不足70行:
function Zhang(options){// 创建构造函数Zhang,并接收对象结构体options
    this.$el=document.querySelector(options.el);// 指定挂载元素
    this.$data=options.data;// 存放你的数据内容
    this.$methods=options.methods;// 存放设你的方法
    this.binding={};// 所有数据相关的订阅者对象都存放于此。最终结构为{数据属性:[订阅者对象,订阅者对象……]}
    this.observer();// 调用观察者,对数据进行劫持
    this.compile(this.$el);// 对元素指令进行解析,订阅者也是在此处创建的
}
Zhang.prototype.observer=function(){// 观察者
    var value="";// 定义用于存放数据属性值的变量value
    for(var key in this.$data){ // 遍历数据对象
        value=this.$data[key];// 对象属性值
        this.binding[key]=[];// 数据订阅者初始化,是一个数组,
        var binding=this.binding[key];// 用于存放本数据相关的所有订阅者,初始为[]
        Object.defineProperty(this.$data,key,{// 开始设置劫持
            get(){
                return value;// 读取值为value
            },
            set(v){// v为设置的值
                if(v!==value){// 当设置的值与当前值不相等时
                    value=v;// 将读取值更新为v
                    binding.forEach(watcher=>{
                        watcher.update();// 通知与本数据相关的订阅者们进行视图更新
                    })
                }
            }
        })
    }
}
Zhang.prototype.compile=function(el){// 解析器
    var nodes=el.children;// 获得所有子节点
    for(var i=0;i<nodes.length;i++){// 对子节点进行遍历
        var node=nodes[i];// 具体节点
        if(node.children.length>0)// 判断是否具有子节点
            this.compile(node);// 如果有子点进行递归操作
        if(node.hasAttribute("z-on:click")){// 该节点是否拥有z-on指令
            var attrVal=node.getAttribute("z-on:click");// 得到指令对应的方法名
            // 为元素绑定click事件,事件方法为$methods下的方法,并将其this指向this.$data
            node.addEventListener("click",this.$methods[attrVal].bind(this.$data))
        }
        if(node.hasAttribute("z-model")){// 该节点是否拥有z-model指令
            var attrVal=node.getAttribute("z-model");// 获得指令对应的数据属性
            node.addEventListener("input",((i)=>{// 为指令添加input事件
                this.binding[attrVal].push(new Watcher(node,"value",this,attrVal));// 为该数据添加订阅者
                return ()=>{
                    this.$data[attrVal]=nodes[i].value;// 更新$data的属性值,会在观察者中进行劫持
                }
            })(i))
        }
        if(node.hasAttribute("z-html")){// 该节点是否拥有z-html指令
            var attrVal=node.getAttribute("z-html");// 获得指令对应的数据属性
            this.binding[attrVal].push(new Watcher(node,"innerHTML",this,attrVal));
        }
        if(node.hasAttribute("z-text")){// 该节点是否拥有z-text指令
            var attrVal=node.getAttribute("z-text");// 获得指令对应的数据属性
            this.binding[attrVal].push(new Watcher(node,"innerText",this,attrVal));
        }
    }
}
function Watcher(el,attr,vm,val){// 观察者
    this.el=el;     // 指令所在的元素
    this.attr=attr;// 绑定的属性名
    this.vm=vm;    // 指令所在实例
    this.val=val;  // 指令的值
    this.update(); // 更新视图view
}
Watcher.prototype.update=function(){
    this.el[this.attr]=this.vm.$data[this.val];
}
—————END—————
喜欢本文的朋友们,欢迎关注公众号 张培跃,收看更多精彩内容
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容

  • 夜很静,但是能隐约听到隔壁的音乐声,这算是房子隔音不好吧!也可以听到自己的心跳声,美中不足的是居然还有一只蚊子在我...
    雪玲珑阅读 157评论 0 1
  • 标签:JS 设计模式 读书笔记链接: 《JS设计模式》读书笔记(一) 《JS设计模式》读书笔记(二) 《JS设计模...
    philoZhe阅读 647评论 0 3
  • 刷圈: 1,产品卖点 2,产品买家真实反馈 3,产品图片 4,产品竞争优势 5,年会招商 6,团队实力 7,系统培...
    zhy800512阅读 259评论 0 1