【设计模式】javascript中的观察者模式

作为事件驱动编程方式的javascript,观察者模式在项目开发中体现了强大的灵活性和在对象一对多的应用场景中占据重要的位置 。

对于javascript来说,这种模式多用于对程序中的某个对象(具体到某个DOM元素或者某个json数据对象)的状态进行观察,并且在其状态发生改变的时候通知它的订阅者

适应场合

在javascript已经内置了许多事件,这些事件大多数是由用户的行为触发的,而应程序的数据状态即是由用户的行为或者其他应用程序逻辑导致了其状态的变化,这些状态就让观察者模式就派上用场了,例如你可以在你正在使用一个table装载的财务月结凭证的数据,此时,并且在本地多分凭证的缓存数据,并且这些凭证的数据是相互关联的,此时我们可以将当前月结凭证作为继承一个实现了Publisher的基类,而其他凭证报表即统一实现一个同一的交互接口,一旦变更的数据项同步到其他凭证当中。

观察者模式的参与成员

  • 发布者(有些书本称“可观察对象”):在计算机世界中,通常是一个具备广播消息的程序代码。
  • 订阅者(有些文章称“观察者”):在现实生活中可以是终端app或者登陆账户。

但上面两者,在js的世界中很灵活它可以是服务端,可以是同一个页面实现同一个应用的其他dom元素,甚者这两个角色可以是同一个继承类的实例。

具体实现

对于发布者当然必须具备前两个基本的功能

  • 被订阅 这里subscribe
  • 被退订 unsubscribe
  • 消息发布 deliver_all

这里,刚开始接触设计的模式的读者,可能会问:“subscribe”不是由订阅者提供的吗?那换个角度想一下,发布者主要提供一个List维持订阅的用户队列,这个subcribe应该是有发布者提供并且实现的,unsubcribe同理。

对于众多订阅(这些订阅者之间可以毫不相关)者必须具备一个统一的应用接口,让发布者知道如何向他们通过该接口发布消息。也就是说,涉及基本两个设计原则,可以做到松耦合:

  • 该接口由发布者倡导(统一接口名称)
  • 由订阅者各自具体实现,发布者不参与接口的任何实现

以下是一个具体的实例


屏幕快照 2019-07-17 下午7.15.09.png

Web测试页

<!DOCTYPE html>
<html lang="en">
<head>
    <title>Async JS</title>
    <link rel="stylesheet" href="{{ url_for('static', filename='css/test.css') }}">
</head>
<body>

<div class="square" id="elementObjserver1">

</div>
<hr>
<button type="button" onclick="model.increment();">Increment</button>
</body>
<script src="{{ url_for('static', filename='js/business/publisher.js') }}"></script>
<script src="{{ url_for('static', filename='js/business/elementObserver.js') }}"></script>
<script src="{{ url_for('static', filename='js/business/consoleObserver.js') }}"></script>
<script src="{{ url_for('static', filename='js/business/historyObserver.js') }}"></script>
<script src="{{ url_for('static', filename='js/business/test.js') }}"></script>

</html>

发布者

class Publisher{
    constructor(){
        this.number=0;
        this.color='红色';
        this.subcribers=[];
    }

    increment(){
        const colors=['橙色','红色','绿色','蓝色','黄色'];

        this.number++;
        this.color=colors[Math.floor(Math.random()*colors.length)];

        this.deliver_all();
    }

    /**
     * 订阅
     * @param o
     */
    subscribe(o){
        this.subcribers.push(o);
    }

    /**
     * 退订
     * @param o
     */
    unsubscribe(o){
        for(let i=0; i<this.subcribers.length; i++){
            if(this.subcribers[i]==o){
                this.subcribers.splice(i,1);
            }
            i--;
        }
    }

    /**
     * 通知所有订阅者
     */
    deliver_all(){
        for(let o of this.subcribers){
            o.update(this);
        }
    }

    /**
     * 通知某个订阅者
     * @param i
     */
    notify_one(i){
        this.subcribers[i].update(this);
    }
}

观察者

可以只要遵循上面说的“统一的交互接口”,这里我们创建了三个毫不相干的观察者对象,但他们都遵循了统一交互接口的update()并且各自有不同的功能实现

class Console{
    update(model){
        console.log('The number is '+model.number+
        ' and the color is '+model.color.toUpperCase());
    }
}

class History{
    constructor(){
        this.colorHistory=[];
    }

    update(model){
        this.colorHistory.unshift(model.color[0].toUpperCase());
        let msg='The most recent 5 color mere:';
        for(let i=0;i<5;i++){

            if(this.colorHistory[i]) {
                msg += this.colorHistory[i] + ' ';
            }
        }
        console.log(msg);
    }
}

class Elem{
    constructor(elementId){
        this.element=document.getElementById(elementId);
    }

    update(model){
        this.element.innerHTML=model.number;
        this.element.style.backgroundColor=model.color;
    }
}

调用测试

const model=new Publisher();
const eObs1=new Elem('elementObjserver1');
const consoleOb=new Console();
const historyOb=new History();

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