• Vue双向绑定实现原理demo


    一.index.html

    <!DOCTYPE html>
    <html>
        <head>
            <meta charset="utf-8" />
            <title>Vue双向绑定原理</title>
            <script src="js/myVue.js"></script>
        </head>
        <body>
            <div id="myApp">
                <input type="text" style="200px" v-model="value">
                <button v-on:click="fn">清空</button>
                <div v-text="value"></div>
                <div v-html="value"></div>
            </div>
        </body>
        <script>
        var vm = new MyVue({
            el: "#myApp",
            data: {
                value: "<h1>VUE</h1>"
            },
            methods: {
                fn() {
                    this.value = "";
                }
            }
        })
    </script>
    </html>

    二.myVue.js

    function MyVue(options){// 创建构造函数MyVue,并接收对象结构体options
        this.$el=document.querySelector(options.el);// 指定挂载元素
        this.$data=options.data;// 存放你的数据内容
        this.$methods=options.methods;// 存放设你的方法
        this.binding={};// 所有数据相关的订阅者对象都存放于此。最终结构为{数据属性:[订阅者对象,订阅者对象……]}
        this.observer();// 调用观察者,对数据进行劫持
        this.compile(this.$el);// 对元素指令进行解析,订阅者也是在此处创建的
    }
    MyVue.prototype.observer=function(){// 观察者
        let value="";// 定义用于存放数据属性值的变量value
        for(let key in this.$data){ // 遍历数据对象
            value=this.$data[key];// 对象属性值
            this.binding[key]=[];// 数据订阅者初始化,是一个数组,
            let 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();// 通知与本数据相关的订阅者们进行视图更新
                        })
                    }
                }
            })
        }
    }
    MyVue.prototype.compile=function(el){// 解析器
        let nodes=el.children;// 获得所有子节点
        for(let i=0;i<nodes.length;i++){// 对子节点进行遍历
            let node=nodes[i];// 具体节点
            if(node.children.length>0)// 判断是否具有子节点
                this.compile(node);// 如果有子点进行递归操作
            if(node.hasAttribute("v-on:click")){// 该节点是否拥有v-on指令
                let attrVal=node.getAttribute("v-on:click");// 得到指令对应的方法名
                // 为元素绑定click事件,事件方法为$methods下的方法,并将其this指向this.$data
                node.addEventListener("click",this.$methods[attrVal].bind(this.$data))
            }
            if(node.hasAttribute("v-model")){// 该节点是否拥有v-model指令
                let attrVal=node.getAttribute("v-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("v-html")){// 该节点是否拥有v-html指令
                let attrVal=node.getAttribute("v-html");// 获得指令对应的数据属性
                this.binding[attrVal].push(new Watcher(node,"innerHTML",this,attrVal));
            }
            if(node.hasAttribute("v-text")){// 该节点是否拥有v-text指令
                let attrVal=node.getAttribute("v-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];
    }

     三.效果图

  • 相关阅读:
    Adobe Creative Cloud for Win 解决:应用程序无法正常启动(0x0000022)。请单击“确定”关闭应用程序
    项目管理 PM 第六章 时间管理
    R for mac安装 无法争取安装Unable to install any packages 安装 r 里的 igraph 报错 ERROR: configuration failed for package ‘igraph’
    Python 相对路径的解决 “Unicode Error ”unicodeescape" codec can't decode bytes… Cannot open text files in Python 3
    GraphML入门教程
    文案——编辑——希腊字母Greek Alphabet
    Tensorflow入门
    安装Tensorflow及安装过程中的错误——ModuleNotFoundError: No module named 'tensorflow'
    基于 Virtualenv 的安装Tensorflow
    小程序--腾讯地图导航
  • 原文地址:https://www.cnblogs.com/qq1272850043/p/10396787.html
Copyright © 2020-2023  润新知