Array的变化侦测(一)

如何追踪变化

为什么对于Array的侦测方式和Object的不同?如下一句push操作,调用的是数组原型上的方法改变数组,不会触发getter/setter。

this.list.push(1);

在ES6之前,JavaScript并没有提供元编程的能力,足以拦截原型方法。Vue的做法是写自定义方法覆盖原型方法。


使用拦截器覆盖原生原型方法.png

用一个拦截器覆盖Array.prototype,每当我们调用原型方法操作数组时,调用的都是自定义方法,就可以跟踪到变化了。

拦截器

拦截器和Array.prototype一样也是一个对象,包含的属性也一样,只是一些能改变数组的方法是处理过的。
整理一下,发现数组原型可以改变数组自身内容的方法有七个:push、pop、shift、unshift、splice、sorte和reverse。

const arrayProto = Array.prototype;
export const arrayMethods = Object.create(arrayProto);
[
    'push',
    'pop',
    'shift',
    'unshift',
    'splice',
    'sorte',
    'reverse'
].forEach(function(method){
    // 缓存原始方法
    const original = arrayProto[method];
    Object.defineProperty(arrayMethods, method, {
        value: function mutator(...args){
            return original.apply(this, args);
        },
        enumerable: false,
        writeable: ture,
        configurable: true
    })
})

这样我们就可以在mutator函数中做一些事情了,比如发送变化的通知。

使用拦截器覆盖Array原型
export class Observer{
    constructor(value){
        this.value = value;
        if(Array.isArray(value)){
            value.__proto__ = arrayMethods;
        } else {
            this.walk(value);
        }
    }
}

__proto__其实是Object.getPrototypeOf和Object.setPrototypeOf的早期实现,只是ES6的浏览器支持度不理想。

使用__proto__覆盖原型.png

将拦截器方法挂载到数组属性上

并不是所有浏览器都支持通过__proto__访问原型,所以还要处理不能使用这个非标准属性的情况。
Vue的做法非常粗暴,直接将arrayMethods身上的方法设置到被侦测数组上。

const hasProto = '__proto__' in {};
const arrayKeys = Object.getOwnPropertyNames(arrayMethods);

export class Observer{
    constructor(value){
        this.value = value;
        if(Array.isArray(value)){
            const augment = hasProto ? protoAugment : copyAugment;
            augment(value, arrayMethods, arrayKeys);
        } else {
            this.walk(value);
        }
    }
}
function protoAugment(target, src, keys){
    target.__proto__ = src;
}
function copyAugment(target, src, keys){
    for(let i = 0, l = keys.length;i < l;i++){
        const key = keys[i];
        def(target, key, src[key]);
    }
}
如何收集依赖

我们创建拦截器实际上是为了获得一种能力,一种感知数组内容发生变化的能力。现在具备了这个能力,要通知谁呢?根据前面对Object的处理,通知Dep中的依赖(Watcher)。
怎么收集依赖呢?还用getter。

function defineReactive(data, key, val){
    if(typeof val = 'object'){
        new Observer(val);
    }
    let dep = new Dep();
    Object.defineProperty(data, key, {
        enumerable: true,
        configurable: true,
        get: function(){
            dep.depend();
            // 这里收集依赖
            return val;
        },
        set: function(newVal){
            if(val === newVal){
                return;
            }
            dep.notify();
            val = newVal;
        }
    })
}

新增了一段注释,也就是说Array在getter中收集依赖,在拦截器触发依赖

依赖收集在哪
export class Observer{
    constructor(value){
        this.value = value;
        this.dep = new Dep(); // 新增dep
        if(Array.isArray(value)){
            const augment = hasProto ? protoAugment : copyAugment;
            augment(value, arrayMethods, arrayKeys);
        } else {
            this.walk(value);
        }
    }
}

Vue将依赖列表存在了Observer,为什么是这里?
前面说Array在getter中收集依赖,在拦截器触发依赖,所以依赖的位置很关键,保证getter要访问的到,拦截器也访问的到。

收集依赖

Dep实例保存在Observer的属性上后,我们开始收集依赖。

function defineReactive(data, key, val){
    let childOb = observe(val); // 修改
    let dep = new Dep();
    Object.defineProperty(data, key, {
        enumerable: true,
        configurable: true,
        get: function(){
            dep.depend();
            
            // 新增
            if(childOb){
                childOb.dep.depend();
            }
            return val;
        },
        set: function(newVal){
            if(val === newVal){
                return;
            }
            dep.notify();
            val = newVal;
        }
    })
}

export function observe(value, asRootData){
    if(!isObject(value)){
        return;
    }
    let ob;
    if(hasOwn(value, '__ob__')&&value.__ob__ instanceof Observer) {
        ob = value.__ob__;
    } else {
        ob = observe(val);
    }
    return ob;
}

增加一个childOb 的意义到底是啥?在于搭建了从getter把依赖收集到Observer的dep中的桥梁。

在拦截器中获取Observer

因为拦截器是对数组原型的封装,所以拦截器可以访问到this(正在被操作的数组)。而dep在Observer中,所以需要在this上读到Observer实例。

// 工具函数
function def(obj, key, val, enumerable){
    Object.defineProperty(obj, key, {
        value: val,
        enumerable: !!enumerable,
        writeable: true,
        configurable: true
    })
}
export class Observer{
    constructor(value){
        this.value = value;

        def(value, '__ob__', this); // 新增
        if(Array.isArray(value)){
            const augment = hasProto ? protoAugment : copyAugment;
            augment(value, arrayMethods, arrayKeys);
        } else {
            this.walk(value);
        }
    }
}

现在Observer实例已经存入数组中__ob__属性,下一步就是在拦截器中获取。

const arrayProto = Array.prototype;
export const arrayMethods = Object.create(arrayProto);
[
    'push',
    'pop',
    'shift',
    'unshift',
    'splice',
    'sorte',
    'reverse'
].forEach(function(method){
    const original = arrayProto[method];
    Object.defineProperty(arrayMethods, method, {
        value: function mutator(...args){
            const ob = this.__ob__; // 新增
            return original.apply(this, args);
        },
        enumerable: false,
        writeable: ture,
        configurable: true
    })
})
向数组的依赖发通知
const arrayProto = Array.prototype;
export const arrayMethods = Object.create(arrayProto);
[
    'push',
    'pop',
    'shift',
    'unshift',
    'splice',
    'sorte',
    'reverse'
].forEach(function(method){
    const original = arrayProto[method];
    Object.defineProperty(arrayMethods, method, {
        value: function mutator(...args){
            const ob = this.__ob__;
            ob.dep.notify(); // 向依赖发通知
            return original.apply(this, args);
        },
        enumerable: false,
        writeable: ture,
        configurable: true
    })
})

既然能获取到Observer实例和里面的依赖列表了,就直接调用notify。

剩下的内容就是获取数组元素变化,以及Vue的处理方式的弊端,另开一篇写吧。

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