Vue2.0 双向绑定的缺陷
vue 实例创建后,无法检测到对象属性的新增或删除,只能追踪到数据是否被修改(Object.defineProperty只能劫持对象的属性),需要使用$set或者forceUpdate
不能监听数组的变化
vue在实现数组的响应式时,它使用了一些hack,把无法监听数组的情况通过重写数组的部分方法来实现响应式,这也只限制在数组的push/pop/shift/unshift/splice/sort/reverse七个方法
vue实现数组响应式的方法
申明一个对象arr,继承array.prototype,然后为其增加了数组的增删改排序和反转的方法,在使用这些方法的时候先调用array.prototype里面原来的方法对其数组进行操作,然后再触发视图更新的方法,在observe时就会先判断该元素是否是对象,是的话就把他的__proto__等于arr
通过重写数组的Array.prototype对应的方法,具体来说就是重新指定要操作数组的prototype,并重新该prototype中对应上面的7个数组方法,通过下面代码简单了解下实现原理:
const methods = ['pop','shift','unshift','sort','reverse','splice', 'push']; // 复制Array.prototype,并将其prototype指向Array.prototype let proto = Object.create(Array.prototype); methods.forEach(method => { proto[method] = function () { // 重写proto中的数组方法 Array.prototype[method].call(this, ...arguments); viewRender() // 视图更新 function observe(obj) { if (Array.isArray(obj)) { // 数组实现响应式 obj.__proto__ = proto; // 改变传入数组的prototype return; } if (typeof obj === 'object') { ... // 对象的响应式实现 } } } })