前端进阶:浅析Vue——手写vue响应原理

Vue 最独特的特性之一,是其非侵入性的响应式系统。数据模型仅仅是普通的 JavaScript 对象。而当你修改它们时,视图会进行更新,这使得状态管理非常简单直接。

先附上HTML代码

  <body id="app">
    <input type="text" k-model="name"/>
    <span k-bind="name"></span>
    <br>
    <input type="text" k-model="age"/>
    <span k-bind="age"></span>
  </body>

废话不说,开撸,一步一步跟着敲实现vue双向绑定

第一步:从函数开始
  function Kve(options) {
    this.$el = document.querySelector(options.el);
    this.$data = {};
    this.$Attr = [];

    this.loop(this.$el)
    
    this.observer(options.data)
  }
  • this.$el:获取所有的DOM元素
  • loop和observer:至关重要的两个方法后续会讲到
第二步:注入灵魂——实例化Kve
  window.onload = function() {
    // 实例化Kve
    var app = new Kve({
      el: '#app',
      data: {
        name: 'moxuanya',
        age: '25'
      },
    })
  }

是不是很眼熟,眼熟就对了,结合第一步,初始化绑定数据

第三部:谁在盯着视图的改变——创建观察者团队
  // 观察者类
  function Watcher(name, el, exp, attr) {
    this.name = name;         // 指令名称
    this.el = el;             // 指令对应的DOM
    this.exp = exp;           // 指令对应的值,如"name"
    this.attr = attr;         // 绑定的属性值,如"textContent"
  }

这一步是抽象的,也是核心思想,需要有人盯着视图,一旦视图改变刷新数据,一旦数据改变刷新视图,暂时不能理解没事,跟着下一步看看观察者们是怎么诞生的

第四步:观察者领证上岗
  // 初始化循环DOM
  Kve.prototype.loop = function(el) {
    var nodes = el.children;
    var _this = this;
    for(var i=0; i<nodes.length-1; i++){
      // 筛选含有指令的DOM节点
      if (nodes[i].hasAttribute('k-model')) {
        nodes[i].addEventListener('input', (function(key) {
          var attrVal = nodes[i].getAttribute('k-model');
          // 添加一个input的观察者     
          _this.$Attr.push(new Watcher('input',nodes[i],attrVal,'value'))
          return function() {
            // 监听input变化,触发Object.definePropert
            _this.$key = attrVal;
            _this.$data[attrVal] =  nodes[key].value;
          }
        })(i)); // 自执行一一绑定input
      }
      if (nodes[i].hasAttribute('k-bind')) {
        var attrVal = nodes[i].getAttribute('k-bind');
        // 添加一个k-bind的观察者
        _this.$Attr.push(new Watcher('span',nodes[i],attrVal,'textContent'))
      }
    }
  }
  • new Watcher():新增一个dom元素的观察者,比如观察input的路人甲,观察span的路人乙
  • _this.$Attr.push:把路人甲,路人乙等等丢进观察者路人团队
  • Watcher 伟大的观察者们,这里的核心思想把视图跟数据绑定在一起
第五步:滴滴!数据改变,行动!
  // 观察者的行为
  Kve.prototype.observer = function(obj) {
    var _this = this;
    for (const key in obj) {
      if (obj.hasOwnProperty(key)) {
        Object.defineProperty(this.$data, key, {
          enumerable: true,
          configurable: true,
          get: function() {
            console.log('get:监听到了值获取')
          },
          set: function(val) {
            console.log('set:监听到了值改变')
            _this.update(key,val)
          }
        })
        // 初始化data
        _this.$data[key] = obj[key];
      }
    }
  }
  // 更新
  Kve.prototype.update = function(key,val) {
    for (const i of this.$Attr) {
      if( i.exp === key){
        i.el[i.attr] = val
      }
    }
  }

Object.defineProperty() 可以监听到数据的改变,这里不做过多解释,一旦数据改变调用update方法,把观察者们叫过来,大家看一看这个值改变的key是哪个观察者手中的key,比如:路人甲把你的input值改了。就这样,一个简单的vue响应原理实例就完成了,博主学艺不深,如有错误还望海涵。

千里之行始于足下,冰冻三尺非一日之寒,加油吧骚年!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容