js观察者模式与Model

目的

观察者模式是常见的设计模式,可以被应用到MV*框架的Model上,来实现对数据变化的监听。

基本概念

观察者模式是一种常见的设计模式。被观察者可以被订阅(subscribe),并在状态发生改变时通知订阅者。

观察者模式的实现主要涉及三个接口:

1. subscribe (evtName, handler):订阅被观察者的指定事件。

2. unsubscribe (evtName, handler):取消对被观察者指定事件的订阅。

3. publish (evtName, data):被观察者发布指定事件。

这些接口也常被命名为on,off,once,trigger。

代码实现

代码很简单易懂,就不多做解释了:

function Observer() {
    this.fns = {};
}
Observer.prototype = {
    subscribe: function (evtName, handler) {
        if (!this.fns[evtName]) {
            this.fns[evtName] = [];
        }
        this.fns[evtName].push(handler);
    },
    unsubscribe: function (evtName, handler) {
        var arr =  this.fns[evtName];
        if (!arr) { return; }
        var index = arr.indexOf(handler);
        if (index !== -1) {
            arr.splice(index, 1);
        }
    },
    publish: function (evtName, data) {
        var arr =  this.fns[evtName];
        if (!arr) { return; }
        arr.forEach(
            function (handler) {
                handler(data);
            }
        );
    }
};

使用起来是这样的:

var benben = new Observer();
benben.subscribe(‘evtA‘, function (data) { console.log(data); });
benben.publish(‘evtA‘, {name: ‘笨笨‘});

与Model结合

接下来我们就要让Model变为可观察的(Observable):

function Model () {
    Observer.call(this); // => this.fns = {};
    this._data = {}; // 我们将实际数据存放在这里
}
$.merge(Model.prototype, Observer.prototype); // 将方法扩展到Model的prototype上

var model = new Model();

现在我们要做的就是,监听model数据的变化,并发布(publish)。方式包括(并不限于):

1. get与set函数:

Model.prototype.set = function (key, value) {
    var oldValue = this._data[key];
    this._data[key] = value;
    this.publish(‘change:‘ + key, {
        key: key,
        value: value,
        oldValue: oldValue
    });
}
Model.prototype.get = function (key) {
    return this._data[key];
}

model.set(‘name‘, ‘笨笨‘);

2. setter与getter:

Model.prototype.defineKeys = function (definitions) {
    var that = this,
        data = this._data;
    for (key in definitions) {
        this[key] = definitions[key];
        this.__defineGetter__(key, function () {
            return data[key];
        });
        this.__defineSetter__(key, function (value) {
            var oldValue = data[key];
            data[key] = value;
            that.publish(‘change:‘ + key, {
                key: key,
                value: value,
                oldValue: oldValue
            });
        });
    }
}

/* 或者使用defineProperty
Object.defineProperty(this, key, {
    get: function() {
        //
    },
    set: function(value) {
        //
    }
});
*/

// 需要先define后赋值
model.defineKeys({name: undefined});
model.name = ‘笨笨‘;

这两种方式都可以将对model的赋值与事件发布绑定到一起。当然它们各自有各自的缺陷:后者的赋值方式更“自然”,但需要先对字段定义。其他的方式还包括数据脏检测(dirty checking)等,但目的是统一的:将model的变化发布给订阅者,比如通知View来更新等等。

小结

通过使用观察者模式,我们就能监听Model的数据变化(也可以reject不符合条件的赋值等等),并作出相应的动作,比如更新View等等。

扩展:ES6(7)中的Object.observe

Object.observe是未来ES标准之一,包括Chrome 36(beta)+的浏览器已经支持之一特性,不过何时标准最终定稿和普遍实现还是未知。

让我们来了解一下Object.observe:

var obj = {name: ‘benben‘},
    arr = [],
    onChange = function (changes) {
        changes.forEach(function (change) {
            console.log(
                change.type, // add, delete, update
                change.object,
                change.name,
                change.oldValue
                );
        });
    };
Object.observe(obj, onChange);
Object.observe(arr, onChange);
obj.name = ‘笨笨‘;
arr.push(1);

试试看,第一个情况我们被通知name的变化,第二种情况则被通知[0]和length发生了变化,是不是很方便呢。

我们还可以指定我们感兴趣的字段,以及取消监听:

Object.observe(obj, onChange, [‘name‘, ‘gender‘]);
Object.unobserve(obj, onChange);

同时还提供了 Object.getNotifier和notifier.notify两个API来帮助我们发布事件:

function Square () {
    this.edge = 0;
}
Square.prototype.setEdge = function (val) {
    var notifier = Object.getNotifier(this);
    this.edge = val;
    notifier.notify({
      object: this,
      type: ‘update‘,
      area: val * val
    });
}

var s = new Square(),
    onChange = function (changes) {
        console.log(‘onChange...‘);
        changes.forEach(function (change) {
            console.log(change);
        });
    };
Object.observe(s, onChange);

s.setEdge(5);

注意s并没有area字段,我们通过notifier的notify方法来发布变化事件。

如果被观察者认为观察者并不应该关注某些字段的变化(不同于观察者只选择观察指定字段集),这是我们可以使用notifier的performChange方法:

Square.prototype.setEdge = function (val) {
    var notifier = Object.getNotifier(this);
    notifier.performChange(‘area‘, function() {
      this.edge = val;
    }, this);
    notifier.notify({
      object: this,
      type: ‘update‘,
      area: val * val
    });
}

我们在performChange的回调中设置了edge的值,这种情况下,edge的变化并不会被发布。

时间: 2024-10-10 20:38:45

js观察者模式与Model的相关文章

Ember.js 入门指南——model简介2

本文接上一篇<Ember.js 入门指南--model简介1>. 2,核心概念 声明:下面简介内摘抄至http://www.emberjs.cn/guides/models/#toc_. 1,store store是应用存放记录的中心仓库.你可以认为store是应用的所有数据的缓存.应用的控制器和路由都可以访问这个共享的store:当它们需要显示或者修改一个记录时,首先就需要访问store. DS.Store的实例会被自动创建,并且该实例被应用中所有的对象所共享. store可以看做是一个缓存

Node.js与Sails~Model和ORM的持久化

回到目录 上一讲说了在sails里定义model及相关参数的说明,这一讲主要说一下如何将你的Model持久化到文件,关系数据库和Nosql数据库里,在持久化这点上,sails是统一管理的,这是优势,也是劣势:优势就是入口统一,用哪种介质持久化,一个设置就OK了:劣势也很明显,就是在一个sails项目里,只能存在一种持久化的方法,你想让mysql和mongodb并存,那是不行的,呵呵! 对于数据的持久化主要分为以下几个步骤,下面一一讲解 1 安装缺失的驱动,默认来说mongodb和sqlserve

JS - Document Object Model

The DOM specifies: how browsers should create a model of an HTML page. how JavaScript can access and update the contents of a web page while it is in the browser window. DOM is implemented by all major browser makers, covers two primary areas: Making

Backbone.js 中使用 Model

前面几篇 Backbone.js 的例子中有使用到 template, 及数据的填充,其实这已经很接近 Model 了.现在来学习怎么创建自己的 Model 类,并简单的使用.Backbone.js 中 Model 会涉及到很多的概念,如 Model 的初始化,默认值,属性的读写,属性值改变的监听,数据渲染,校验,以及与服务端的数据同步等. 本文不打算讲前面大部分的内容,最初思考的一个脉络是怎么把 Model 引入进来,所以线索会是 建立 Model 类 -> 初始化 Model 实例 ->

Backbone.js学习之Model

首先,我们看一下官方文档中对Model的解释(或者说定义): Models are the heart of any JavaScript application, containing the interactive data as well as a large part of the logic surrounding it: conversions, validations, computed properties, and access control. You extend Back

js 观察者模式

观察者模式定义一种一对多的关系,多个观察者订阅一个主题对象,当主题对象发生改变的时候通知所有观察者,使他们能实现更新. 具体实现方式: 定义一个被观察对象,定义一个主题数组,观察者订阅主题通过往数组对象内添加回调函数来实现,当主题发生改变时轮流调用回调函数. var subpub= { topics: [], uid: 0, //订阅 subscribe: function(topic, callback){ var tmp= this.topics[topic] || [] tmp.push(

js观察者模式学习

function Events(){ var obj = {}; this.on=function(key,fn){ var stack; stack = obj[key] || (obj[key] =[]); return stack.push(fn); } this.off=function(key){ var stack; return (stack=obj[key]) != null ? stack.length=0 : void 0;//此处stack.length=0不等同于stac

js观察者模式

var pubsub = (function(){ var q = {} topics = {}, subUid = -1; //发布消息 q.publish = function(topic, args) { if(!topics[topic]) {return;} var subs = topics[topic], len = subs.length; while(len--) { subs[len].func(topic, args); } return this; }; //订阅事件 q

JS实现观察者模式(订阅/发布模式)

实现 /*  * js 观察者模式 又称 订阅/发布模式  * 通过创建"可观察"对象,当发生一个感兴趣的事件时可将该事件通告给  * 所有观察者,从而形成松耦合 */ // 通用的发布者 EventPublisher = Base.extend({ publish: function(data, type) { EventPublisher.publish(data, type); } }, { subscribers : {         any : []    // 事件类型: