记录简单实现Vue2中的响应式原理

Vue是数据驱动视图模式,数据变更后,会自动更新视图。

原生js中要更新视图通常是通过触发事件,然后修改数据,最后获取dom节点并且把数据更新到dom节点。

在Vue的开发中我们只用修改数据就可以达到视图更新的效果,<b>这里Vue通过数据的监听帮助我们实现了视图的更新</b>,这是Vue中的响应式。

所以要实现响应式的原理(简单实现~)要解决两个问题:

1、监听数据的变化。

2、数据发生变化时更新视图。

本文先解决第一个问题:如何监听数据的变化。

这里要用到 ES5 中的 Object.defineProperty() 方法。查询 MDN 可以看到该方法的作用会直接在对象上定义一个新属性,或者修改一个对象的现有属性,并返回此对象。其语法如下:

    Object.defineProperty(obj, prop, descriptor)

其中参数 obj 是要定义的对象(目标对象),prop 是要定义或修改的属性名称(key),descriptor 是属性描述符。

说一句题外话。Vue这里用到了ES5的Object.defineProperty()方法,也说明Vue不兼容IE8及其以下的浏览器。

<b>这里重点是属性描述符,它是一个对象类型的数据,其包含很多属性,这里重点用到的是其 get 和 set 属性。</b>

MDN 中关于 get 和 set 的解释如下:

get:属性的getter函数,当访问该属性时,会调用此函数,函数不接受参数,函数的返回值会被用作属性的值。默认为 <b>undefined </b>

通俗讲的意思就是,当调用对象中某个属性时,会触发该属性上的get方法,并且该getter的返回值会当做时该属性的值。




set:属性的setter函数,当属性值被修改时,会调用该方法,此方法接受一个参数,这个参数就是该属性被赋予的新值。默认为 <b>undefined</b>

意思就是当给对象中某个属性赋值时,会触发该属性的set方法,并且将赋予的新值作为参数传递

好了,看到这里应该大概有想法怎么去实现监听数据的变化了。因为属性被赋予新值时会触发该属性上的set方法,所有主要就是去重写属性的set方法,在set方法中调用视图更新。

直接上代码

// 定义数据
var data = {
  name: 'ABC',
  age: 18,
  friends: [1, 2, 3, 4],
  skill: {
    work: 15
  }
}

// 视图更新
const renderView = () => {
  console.log("******* 视图更新 *******", JSON.stringify(data));
}

// 数组类型的数据修改监听 -- 去修改其push\shift\pop等方法,在这些方法中实现视图的更新
const arrProto = Object.create(Array.prototype) // 备份数组原型,因为待会儿会修改数组数组其原型上的方法
const methodList = ['push', 'pop', 'shift']

methodList.forEach(method => {
  arrProto[method] = function () {
    console.log("method: ", method);
    Array.prototype[method].call(this, ...arguments)  // 例用Array原型原本的方法实现数据逻辑
    renderView()  // 实现视图的更新
  }
})

// 实现监听到数据变化的处理方法
const reactive = (obj, key, value) => {
  observer(value) // 迭代,监听对象数据
  Object.defineProperty(obj, key, {
    get() {
      return value
    },
    set(newVal) {
      value = newVal
      console.log("***********key: ", key)
      renderView()
    }
  })
}

// 数据监听方法
const observer = (target) => {
  if (typeof target !== 'object' || target == 'null') {
    return target
  }

  // 数组处理
  if (Array.isArray(target)) {
    target.__proto__ = arrProto // 修改目标数组的原型指向
  } else {
      // 普通对象处理
    for (var key in target) {
      reactive(target, key, target[key])
    }
  }

}

observer(data)
// 手动修改数据,查看控制台打印
// data.age = 20
// data.skill.work = 20

data.friends.push(1)
// data.friends.unshift(1)  // unshift 没有写在methodList中,所以执行这句代码不会触发视图改变

需要说的是,列表类型的数据需要将将视图更新操作放到其原型上的方法中实现。总所周知,Vue 中数组数据是不会响应式更新数组下标赋值的,只有使用 push、pop 等数组方法才会更新视图。

    arr[1] = "A"    // 视图不更新
    arr.push("A" ) // 视图更新

所以对于列表类型的数据需要单独处理,具体操作是改变该数组的原型指向,重新实现其 push、pop 等方法,这些方法原本逻辑的实现依靠 JS 中数组原型 Array.prototype 中对应的方法实现,然后再添加视图更新操作即可。具体就是上面代码中的这个部分:

// 数组类型的数据修改监听 -- 去修改其push\shift\pop等方法,在这些方法中实现视图的更新
const arrProto = Object.create(Array.prototype) // 备份数组原型,因为待会儿会修改数组数组其原型上的方法
const methodList = ['push', 'pop', 'shift']

methodList.forEach(method => {
  arrProto[method] = function () {
    console.log("method: ", method);
    Array.prototype[method].call(this, ...arguments)  // 例用Array原型原本的方法实现数据逻辑
    renderView()  // 实现视图的更新
  }
})

注意,这里只在 methodList 中添加了 push、pop、shift 方法,所以上面代码也只能实现上面三个方法的响应式监听,正常的应该把所有的方法都写在 methodList 里。

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