vue 源码分析 双向数据binding的方法

我们的目的是:更新数据->自动更新ui 1.首要检测数据有没有变化 2.更新ui

根据 vue 的思想, es5 提供我们一个方法,为一个对象某一个属性设置监听方法。

    var data = {
          a: 5;
    }
    
     //我们设置了 data对象的a属性的监听
      Object.defineProperty(data, a, { 
          set: function(newVal) {
                  console.log(newVal,'为data.a设置了新值');
          },
          get: function() {
               console.log('获取data.a');
                return variable; 
          }
       )

    data.a = 10;//会调用set方法
    //打印:10为data.a设置了新值。
    var c = data.a;//会调用gat方法
    //打印:获取data.a
复制代码
  • 吐槽一下,这个Swift语法一模一样啊, 但是 Swift 除了 set 和 gat 方法以外还有 willSet 即将改变和 didSet 改变后的方法。

  • 这是 vue 官方的源码地址 https://github.com/vuejs/vue/blob/871ed9126639c9128c18bb2f19e6afd42c0c5ad9/explorations%2Fgetset.html

###我把源码稍微修改了(基本没动),添加了一些注释。 Chrome 可以直接粘贴运行。

  1. 先用正则把{{}}这里面的数据找到。 以属性为“data-element-binding”绑定到DOM树上
  2. 根据DOM树上“data-element-binding”,找到当前的DOM。
  3. 根据DOM里的参数与data里的属性,通过defineProperty绑定。
  4. 更新data数据,然后DOM自动更新了。
<!DOCTYPE html>
<html>

<head>
    <title>ideal</title>
    <meta charset="utf-8">
</head>

<body>
    <div id="test">
        <p>{{msg}}</p>
        <p>{{msg}}</p>
        <p>{{msg}}</p>
        <p>{{what}}</p>
        <p>{{hey}}</p>
    </div>
    <script>
        var bindingMark = 'data-element-binding'

        function Element(id, initData) {
            var el = document.getElementById(id)
            var bindings = {} //内部暂存绑定数据及dom
            var data = {} //存储bingding数据并实现监控
            var content = el.innerHTML.replace(/\{\{(.*)\}\}/g, markToken) //让{{msg}} 换成 <span data-element-binding="msg"></span>
            el.innerHTML = content //复制回去

            for (var variable in bindings) {
                bind(variable); //将每个数据的名称比如'msg'绑定到data
            }
            if (initData) {
                for (var variable in initData) {
                    data[variable] = initData[variable]
                }
            }

            function markToken(match, variable) { //遍历  match {{msg}}  variable 'msg'
                bindings[variable] = {} //bindings里存储了数据来源的字段比如bindings['msg']
                return '<span ' + bindingMark + '="' + variable + '"></span>'
            }

            function bind(variable) {
                bindings[variable].els = el.querySelectorAll('[' + bindingMark + '="' + variable + '"]'); //bindings里再存储msg绑定的元素

                [].forEach.call(bindings[variable].els, function(e) { //删除data-element-binding属性
                    e.removeAttribute(bindingMark);
                })

                Object.defineProperty(data, variable, { //es5观察属性
                    set: function(newVal) {
                        console.log(variable);
                        [].forEach.call(bindings[variable].els, function(e) {
                            bindings[variable].value = newVal; //=>这里才是实现的绑定,更新内部暂存数据, 为了以后取
                            e.textContent = newVal; //更新数据到dom
                        })
                    },
                    get: function() {
                        return bindings[variable].value; //取数据仅仅是内部暂存的数据
                    }
                })
            }
        }

        var app = new Element('test', {
            msg: 'hello',
            what: 'hi'
        })
    </script>
</body>

</html>


复制代码

个人博客: www.liangtongzhuo.com

转载于:https://juejin.im/post/5a31f136f265da432b4ad564

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值