一个简单的vue双向绑定实现案例

2018-09-05  本文已影响0人  Mr丶Sunny

最近面试经常会问这个问题,你知道vue的双向绑定是通过什么实现的吗?了解的同学应该知道vue是使用Object.defineProperty属性,重写data的get和set方法来实现的。先引用网上的一张图,那么接下来我们就按照这张图的步骤去用代码实现这个功能。


image

先看DOM结构部分,结构部分很简单

<div id="app">
    <input type="text" v-model="number">
    <input type="button" value="增加" v-click="increment"/>
    <input type="button" value="减少" v-click="subtract">
    <h3 v-bind="number"></h3>
</div>

然后就是js部分了,按照上图的步骤我们定义一个构造函数,并且init这个构造函数

    //初始化构造函数
    function Vm(options) {
        this._init(options);
    }

    Vm.prototype._init = function (options) {
        this.$options = options;    //options为上面使用时传入的结构体,包括el、data、methods
        this.$el = document.querySelector(options.el);     //el是#app,this.$el是id为app的Element元素
        this.$data = options.data;      //this.$data = {number: 0 ...}
        this.$methods = options.methods;        //this.$methods = {increment: function(){} ...}

        this._binding = {};     //_binding保存着model与view的映射关系,也就是我们前面定义的Watcher的实例。当model改变时,会触发其中的指令类更新,保证view也能实时更新
    }

这个时候我们还需要给实例添加一个observer、compile方法,observer方法实现对数据的劫持,compile方法负责编译解析指令并且初始化视图

    //实现_observer函数,对data进行处理,重写data的set和get函数
    Vm.prototype._observer = function (data) {
        //如果data数据为空
        if (!data) {
            return;
        }
        if (typeof data !== "object") {
            throw new Error("data必须是一个对象");
        }

        var self = this;
        /*
        * 遍历data中所有属性
        * Object.keys(obj)函数返回一个由一个给定对象的自身可枚举属性组成的数组
        * */
        Object.keys(data).forEach(function (key) {
            //当前属性的值
            var oldValue = data[key];

            /*
            *   按照前面的数据
            *   _binding = {
            *       number:{
            *           _directives: [...watch实例]
            *       }
            *   }
            *   这里是先声明一个空数组,以后所有和data中某值有关系的都会追加到相应的_directives中。
            *   为何要在这里声明?
            *   因为这里要根据_binding的key必须为data中的属性(键)
            *   那这个数组何时才有东西呢?
            *   解析指令(比如v-bind)的时候push进去的,因为解析指令的时候,会解析每一个dom,然后解析出你的是点击事件还是修改文本内容。
            *   然后在生成watcher追加进去这个数组-以后所有和data中某值发生改变只要循环执行这个数组中watcher的update更新方法就可以更新所有和这个data中的某值相关联的dom
            * */
            self._binding[key] = {
                _directives: []
            }
            //获取本data某属性对应的_directives
            var binding = self._binding[key];

            /**
             * 语法:Object.defineProperty(obj, key, descriptor)
             *      @param: obj:需要定义属性的对象;
             *              key:需要定义或修改的属性;
             *              descriptor:将被定义或修改属性的描述符
             */
            Object.defineProperty(data, key, {    //实现双向绑定的关键代码
                enumerable: true, // 可枚举--可被for-in和Object.keys()枚举。
                configurable: true, //当且仅当值为true时,该属性描述符才能够被改变,也能被删除
                //value: undefined,   //该属性对应的值。可以是任何有效的 JavaScript 值(数值,对象,函数等)。默认为 undefined
                //writable: false,    //当且仅当该属性的writable为true时,value才能被赋值运算符改变。默认为 false
                get: function () {  //一个给属性提供getter的方法,当访问该属性时方法会被执行,执行时没有参数传入,但会传入this对象
                    //发现这个oldValue如果替换成data[key]会造成堆栈溢出。
                    return oldValue;
                },
                set: function (newValue) {  //一个给属性提供setter的方法,当属性值修改时触发该方法,该方法将接受唯一参数,即该属性新的参数值。
                    if (oldValue == newValue) return;
                    console.log("监听到值变化了");
                    //发现这个oldValue如果替换成data[key]会造成堆栈溢出。
                    oldValue = newValue;
                    // 当data中某属性改变时,触发_binding['某值']._directives 中的绑定的Watcher类的更新--这样实现一个data中的值发生改变,和它相关联的dom更新。
                    binding._directives.forEach(function (item) {
                        item.update();
                    })
                }
            });
        })
    }

    //定义complie函数,用来解析指令(v-bind,v-model,v-click)等,并在这个过程中对view与model进行绑定。
    Vm.prototype._complie = function (root) {    //root为id为app的ELement元素,也就是vue的根元素
        var _this = this;
        var nodes = root.children;
        for (var i = 0; i < nodes.length; i++) {    //对所有的元素进行遍历,并处理
            var node = nodes[i];
            if (node.children.length) {
                this._complie(node);
            }

            if (node.hasAttribute("v-click")) { //如果有v-click属性,我们监听onclick事件,触发increment、subtract方法
                node.onclick = (function () {
                    var attrVal = nodes[i].getAttribute("v-click");
                    return _this.$methods[attrVal].bind(_this.$data); //bind是使data的作用域与method函数的作用域保持一致
                })();
            }

            if (node.hasAttribute("v-model") && (node.tagName == "INPUT" || node.tagName == "TEXTAREA")) {//如果有v-model属性,并且元素为input或者textarea,我们监听它的input事件
                node.addEventListener("input", (function (key) {
                    var attrVal = node.getAttribute("v-model");
                    /**
                     * _this._binding["number"]._directives = [一个Watcher实例]
                     * 其中Watcher.prototype.update = function() {
                     *     node["value"] = _this.$data["number"]; 这就将node的值保持与number一致
                     * }
                     */
                    _this._binding[attrVal]._directives.push(new Watcher(
                        "input",
                        node,
                        _this,
                        attrVal,
                        "value",
                    ))

                    return function () {
                        _this.$data[attrVal] = nodes[key].value; //使number的值与node的value保持一致,实现双向绑定
                    }
                })(i));
            }

            if (node.hasAttribute("v-bind")) {  //如果有v-bind属性,只要使node的值及时更新为data中number的值即可
                var attrVal = node.getAttribute("v-bind");
                _this._binding[attrVal]._directives.push(new Watcher(
                    'text',
                    node,
                    _this,
                    attrVal,
                    "innerHTML"
                ))
            }
        }
    }

方法定义完了当然得调用,回到init方法

Vm.prototype._init = function (options) {
        this.$options = options;    //options为上面使用时传入的结构体,包括el、data、methods
        this.$el = document.querySelector(options.el);     //el是#app,this.$el是id为app的Element元素
        this.$data = options.data;      //this.$data = {number: 0}
        this.$methods = options.methods;        //this.$methods = {increment: function(){}}

        this._binding = {};     //_binding保存着model与view的映射关系,也就是我们前面定义的Watcher的实例。当model改变时,会触发其中的指令类更新,保证view也能实时更新
        this._observer(this.$data);
        this._complie(this.$el);
    }

接下来实现一个Watcher,用来绑定update方法,实现对DOM的更新

    //实现一个指令类Watcher,用来绑定更新函数,实现对DOM元素的更新
    function Watcher(name, el, vm, exp, attr) {
        this.name = name;       //指令名称,例如文本节点,该值设置为"text"
        this.el = el;           //指令对应的DOM元素
        this.vm = vm;           //指令所属的实例
        this.exp = exp;         //指令对应的值,本例为:"number"
        this.attr = attr;       //指令绑定的属性值,本例为:"innerHTML"

        this.update();
    }

    Watcher.prototype.update = function () {
        this.el[this.attr] = this.vm.$data[this.exp]; //比如H3.innerHtml = this.data.number;当number改变时会触发update函数,保证对应的DOM内容进行更新
    }

调用也很简单,也是vue最熟悉的调用方式

window.onload = function () {
        var vm = new Vm({
            el: "#app",
            data: {
                number: 0,
                age: 18
            },
            methods: {
                increment: function () {
                    this.number++;
                },

                subtract: function () {
                    this.number--;
                }
            }
        })
    }

完整代码如下:

<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div id="app">
    <input type="text" v-model="number">
    <input type="button" value="增加" v-click="increment"/>
    <input type="button" value="减少" v-click="subtract">
    <h3 v-bind="number"></h3>
</div>
</body>
</html>
<script>
    //初始化构造函数
    function Vm(options) {
        this._init(options);
    }

    Vm.prototype._init = function (options) {
        this.$options = options;    //options为上面使用时传入的结构体,包括el、data、methods
        this.$el = document.querySelector(options.el);     //el是#app,this.$el是id为app的Element元素
        this.$data = options.data;      //this.$data = {number: 0}
        this.$methods = options.methods;        //this.$methods = {increment: function(){}}

        this._binding = {};     //_binding保存着model与view的映射关系,也就是我们前面定义的Watcher的实例。当model改变时,会触发其中的指令类更新,保证view也能实时更新
        this._observer(this.$data);
        this._complie(this.$el);
    }

    //实现_observer函数,对data进行处理,重写data的set和get函数
    Vm.prototype._observer = function (data) {
        //如果data数据为空
        if (!data) {
            return;
        }
        if (typeof data !== "object") {
            throw new Error("data必须是一个对象");
        }

        var self = this;
        /*
        * 遍历data中所有属性
        * Object.keys(obj)函数返回一个由一个给定对象的自身可枚举属性组成的数组
        * */
        Object.keys(data).forEach(function (key) {
            //当前属性的值
            var oldValue = data[key];

            /*
            *   按照前面的数据
            *   _binding = {
            *       number:{
            *           _directives: [...watch实例]
            *       }
            *   }
            *   这里是先声明一个空数组,以后所有和data中某值有关系的都会追加到相应的_directives中。
            *   为何要在这里声明?
            *   因为这里要根据_binding的key必须为data中的属性(键)
            *   那这个数组何时才有东西呢?
            *   解析指令(比如v-bind)的时候push进去的,因为解析指令的时候,会解析每一个dom,然后解析出你的是点击事件还是修改文本内容。
            *   然后在生成watcher追加进去这个数组-以后所有和data中某值发生改变只要循环执行这个数组中watcher的update更新方法就可以更新所有和这个data中的某值相关联的dom
            * */
            self._binding[key] = {
                _directives: []
            }
            //获取本data某属性对应的_directives
            var binding = self._binding[key];

            /**
             * 语法:Object.defineProperty(obj, key, descriptor)
             *      @param: obj:需要定义属性的对象;
             *              key:需要定义或修改的属性;
             *              descriptor:将被定义或修改属性的描述符
             */
            Object.defineProperty(data, key, {    //实现双向绑定的关键代码
                enumerable: true, // 可枚举--可被for-in和Object.keys()枚举。
                configurable: true, //当且仅当值为true时,该属性描述符才能够被改变,也能被删除
                //value: undefined,   //该属性对应的值。可以是任何有效的 JavaScript 值(数值,对象,函数等)。默认为 undefined
                //writable: false,    //当且仅当该属性的writable为true时,value才能被赋值运算符改变。默认为 false
                get: function () {  //一个给属性提供getter的方法,当访问该属性时方法会被执行,执行时没有参数传入,但会传入this对象
                    //发现这个oldValue如果替换成data[key]会造成堆栈溢出。
                    return oldValue;
                },
                set: function (newValue) {  //一个给属性提供setter的方法,当属性值修改时触发该方法,该方法将接受唯一参数,即该属性新的参数值。
                    if (oldValue == newValue) return;
                    console.log("监听到值变化了");
                    //发现这个oldValue如果替换成data[key]会造成堆栈溢出。
                    oldValue = newValue;
                    // 当data中某属性改变时,触发_binding['某值']._directives 中的绑定的Watcher类的更新--这样实现一个data中的值发生改变,和它相关联的dom更新。
                    binding._directives.forEach(function (item) {
                        item.update();
                    })
                }
            });
        })
    }

    //定义complie函数,用来解析指令(v-bind,v-model,v-click)等,并在这个过程中对view与model进行绑定。
    Vm.prototype._complie = function (root) {    //root为id为app的ELement元素,也就是vue的根元素
        var _this = this;
        var nodes = root.children;
        for (var i = 0; i < nodes.length; i++) {    //对所有的元素进行遍历,并处理
            var node = nodes[i];
            if (node.children.length) {
                this._complie(node);
            }

            if (node.hasAttribute("v-click")) { //如果有v-click属性,我们监听onclick事件,触发increment、subtract方法
                node.onclick = (function () {
                    var attrVal = nodes[i].getAttribute("v-click");
                    return _this.$methods[attrVal].bind(_this.$data); //bind是使data的作用域与method函数的作用域保持一致
                })();
            }

            if (node.hasAttribute("v-model") && (node.tagName == "INPUT" || node.tagName == "TEXTAREA")) {//如果有v-model属性,并且元素为input或者textarea,我们监听它的input事件
                node.addEventListener("input", (function (key) {
                    var attrVal = node.getAttribute("v-model");
                    /**
                     * _this._binding["number"]._directives = [一个Watcher实例]
                     * 其中Watcher.prototype.update = function() {
                     *     node["value"] = _this.$data["number"]; 这就将node的值保持与number一致
                     * }
                     */
                    _this._binding[attrVal]._directives.push(new Watcher(
                        "input",
                        node,
                        _this,
                        attrVal,
                        "value",
                    ))

                    return function () {
                        _this.$data[attrVal] = nodes[key].value; //使number的值与node的value保持一致,实现双向绑定
                    }
                })(i));
            }

            if (node.hasAttribute("v-bind")) {  //如果有v-bind属性,只要使node的值及时更新为data中number的值即可
                var attrVal = node.getAttribute("v-bind");
                _this._binding[attrVal]._directives.push(new Watcher(
                    'text',
                    node,
                    _this,
                    attrVal,
                    "innerHTML"
                ))
            }
        }
    }

    //实现一个指令类Watcher,用来绑定更新函数,实现对DOM元素的更新
    function Watcher(name, el, vm, exp, attr) {
        this.name = name;       //指令名称,例如文本节点,该值设置为"text"
        this.el = el;           //指令对应的DOM元素
        this.vm = vm;           //指令所属的实例
        this.exp = exp;         //指令对应的值,本例为:"number"
        this.attr = attr;       //指令绑定的属性值,本例为:"innerHTML"

        this.update();
    }

    Watcher.prototype.update = function () {
        this.el[this.attr] = this.vm.$data[this.exp]; //比如H3.innerHtml = this.data.number;当number改变时会触发update函数,保证对应的DOM内容进行更新
    }

    window.onload = function () {
        var vm = new Vm({
            el: "#app",
            data: {
                number: 0,
                age: 18
            },
            methods: {
                increment: function () {
                    this.number++;
                },

                subtract: function () {
                    this.number--;
                }
            }
        })
    }

</script>
上一篇 下一篇

猜你喜欢

热点阅读