BaiDuIFE ------ vue动态数据绑定实现过程

感谢百度前端技术学院带领我进步!谨以此文,记录我的学习过程~
动态数据绑定

Vue的初学者一定对于数据的动态绑定并不陌生,从最简单的需求开始一步步理解其实现原理,以及相关涉及到的知识点。

let app1 = new Observer({
  name: 'youngwind',
  age: 25
});

let app2 = new Observer({
  university: 'bupt',
  major: 'computer'
});

// 要实现的结果如下:
app1.data.name // 你访问了 name
app.data.age = 100;  // 你设置了 age,新的值为100
app2.data.university // 你访问了 university
app2.data.major = 'science'  // 你设置了 major,新的值为 science

实际上这需要用到ES5中的Object.prototype.defineProperty(参见文档)这个方法,简单叙述一下其参数:
<blockquote>
Object.defineProperty(obj,key,descriptor)

  • obj是待操作的对象;
  • key是对象中待操作的属性名
  • descriptor是一个配置对象,其中有:
    configurable:配置总开关,默认为false,只有当其值为true时,才能动态的修改配置方法;
    enumerable:枚举开关,默认为false,只有当其值为true时,该属性才能被枚举;
    value:默认为undefined,即为本意obj[key] = value;
    writable:默认为false,只有其值为true时,属性值才能被改写;
    set/get:为属性提供getter和setter方法。
    </blockquote>

为了实现第一个需求,我们需要为每一个属性均提供getter/setter:

function Observer(data) {
    this.data = data;
    this.makeObserver(data);
}
Observer.prototype.makeObserver = function(data){
   if(typeof data !== "object"){
        throw "please input object!"
    }
    let val;
    //在对象中遍历,只能用for..in..但是这个方法会将原型链上的属性方法均会遍历
    //因此用Object.hasOwnProperty进行过滤,只保留自身对象上的
    for(let key in data){
        if(data.hasOwnProperty(key)){
            val = data[key];
            //如果还是引用类型,则迭代直至所有的属性均绑定了get/set
            if(typeof val === "object"){
                new Observer(val)
            }
            this.convert(key,val);
        }
    }
}
Observer.prototype.convert = function(){
    Object.defineProperty(this.data,key,{
        enumerable:true,
        configurable:true,
        get:function () {
            console.log("你访问了" + key);
            return val;
        },
        set:function (newVal,func) {
            console.log("你设置了" + key + "新的值为" + newVal);
            if(newVal == val) return;
            //如果值为对象,那么还得给对象里的属性绑定setter/getter
            if(typeof newVal == "object"){
                new Observer(newVal);
            }
            val = newVal;
            return val;
        }
    })
}

绑定了getter,setter后,只是进行了数据获取/修改后的反馈,并没有涉及到触发数据改动后的回调。我们可以用事件触发的思路,也就是利用发布订阅(观察者)模式来进行函数回调。
简单来说:就是单独创建一个中间层,用以统一管理事件,该中间层给出两个接口,一个用于订阅事件,一个用于发布事件;
一个简单的实现:

//发布订阅模式,一个中间层(包括一个订阅接口,一个取消接口,一个发布接口)
function PubSub(){
    this.handlers = {};
}
PubSub.prototype.on = function (eventType,handler) {
    if (!(eventType in this.handlers)){
        this.handlers[eventType] = [];
    }
    this.handlers[eventType].push(handler);
    return this;
}
PubSub.prototype.emit = function (eventType) {
    if(!this.handlers[eventType]) return;
    var handlerArgs = [].slice.call(arguments,1);  //刨除eventType,保留其他参数
    for(var i = 0; i < this.handlers[eventType].length;i++){
        this.handlers[eventType][i].apply(this,handlerArgs);  //实际上等于func(..rest);
    }
    return this;
}
PubSub.prototype.off = function (eventType) {
    if(!(eventType in this.handlers)) return;
    delete this.handlers[eventType];
    return this;
}
Observer.prototype.$watch = function(attr, callback){
    this.eventBus.on(attr, callback);
};

第二个需求变为:

 let app1 = new Observer({
         name: 'youngwind',
         age: 25
 });

 // 你需要实现 $watch 这个 API
 app1.$watch('age', function(age) {
         console.log(`我的年纪变了,现在已经是:${age}岁了`)
 });

 app1.data.age = 100; // 输出:'我的年纪变了,现在已经是100岁了'

此时,我们在第一个需求的基础上,利用观察者模式的思想添加中间层:

function Observer(data) {
    this.data = data;
    this.makeObserver(data);
    this.eventBus = new PubSub();
}
//该方法就是给属性绑get/set
Observer.prototype.makeObserver = function (data) {
    if(typeof data !== "object"){
        throw "please input object!"
    }
    let val;
    //在对象中遍历,只能用for..in..但是这个方法会将原型链上的属性方法均会遍历
    //因此用Object.hasOwnProperty进行过滤,只保留自身对象上的
    for(let key in data){
        if(data.hasOwnProperty(key)){
            val = data[key];
            //如果还是引用类型,则迭代直至所有的属性均绑定了get/set
            if(typeof val === "object"){
                new Observer(val)
            }
            this.convert(key,val);
        }
    }
};
Observer.prototype.convert = function (key,val) {
    var that = this;
    Object.defineProperty(this.data,key,{
        enumerable:true,
        configurable:true,
        get:function () {
            console.log("你访问了" + key);
            return val;
        },
        set:function (newVal,func) {
            console.log("你设置了" + key + "新的值为" + newVal);
            if(newVal == val) return;
            if(typeof newVal == "object"){
                new Observer(newVal);
            }
            that.eventBus.emit(key,newVal);    //触发订阅
            val = newVal;
            return val;
        }
    })
};


//发布订阅模式,一个中间层(包括一个订阅接口,一个取消接口,一个发布接口)
function PubSub(){
    this.handlers = {};
}
PubSub.prototype.on = function (eventType,handler) {
    if (!(eventType in this.handlers)){
        this.handlers[eventType] = [];
    }
    this.handlers[eventType].push(handler);
    return this;
}
PubSub.prototype.emit = function (eventType) {
    if(!this.handlers[eventType]) return;
    var handlerArgs = [].slice.call(arguments,1);  //刨除eventType,保留其他参数
    for(var i = 0; i < this.handlers[eventType].length;i++){
        this.handlers[eventType][i].apply(this,handlerArgs);  //实际上等于func(..rest);
    }
    return this;
}
PubSub.prototype.off = function (eventType) {
    if(!(eventType in this.handlers)) return;
    delete this.handlers[eventType];
    return this;
}
Observer.prototype.$watch = function(attr, callback){
    this.eventBus.on(attr, callback);
};


let app1 = new Observer({
    name: 'youngwind',
    age: 25
});


// 你需要实现 $watch 这个 API
app1.$watch('age', function(age) {
    console.log(`我的年纪变了,现在已经是:${age}岁了`)
});

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

推荐阅读更多精彩内容