观察者模式
对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。
观察者要素
1.抽象主题(Subject)角色:把所有对观察者对象的引用保存在一个集合中,每个抽象主题角色都可以有任意数量的观察者。抽象主题提供一个接口,可以增加和删除观察者角色。一般用一个抽象类和接口来实现。
2.抽象观察者(Observer)角色:为所有具体的观察者定义一个接口,在得到主题的通知时更新自己。
3.具体主题(ConcreteSubject)角色:在具体主题内部状态改变时,给所有登记过的观察者发出通知。具体主题角色通常用一个子类实现。
4.具体观察者(ConcreteObserver)角色:该角色实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态相协调。通常用一个子类实现。如果需要,具体观察者角色可以保存一个指向具体主题角色的引用。
例子
为了生产的安全,你决定为你的工厂安装火警报警器,现在你需要为这个报警器写个控制软件。
// 观察者模式
var DP = require("./DesignPattern.js");
function Factory() {
DP.Interface(this, ['attach', 'detach', 'nofityObserver', 'fire'])
//添加火灾报警器
//移除火灾报警器
//通知报警器
//发生火灾
}
function FireAlarm(name) {
var _name=name
this.rang=function(){
console.log(_name+':发生工厂火灾了,鸣笛');
}
}
function PorscheFactory() {
this.__proto__ = new Factory();
var _alarms = [];
this.attach = function (alarm) {
_alarms.push(alarm);
}
this.detach = function (alarm) {
_alarms.splice(_alarms.indexOf(alarm),1);
}
this.nofityObserver = function () {
_alarms.forEach(function(alarm){
alarm.rang()
})
}
this.fire=function(){
console.log('工厂着火了');
this.nofityObserver();
}
}
var f=new PorscheFactory();
var fireAlarm1=new FireAlarm('门卫报警器');
var fireAlarm2=new FireAlarm('消防局报警器');
var fireAlarm3=new FireAlarm('门卫报警器');
f.attach(fireAlarm1);
f.attach(fireAlarm2);
f.attach(fireAlarm3);
f.fire();
console.log('--------------------------------------');
f.detach(fireAlarm3);
f.fire();
观察者模式优点:
1.观察者和被观察者是抽象耦合的。
2.建立一套触发机制。
观察者模式缺点:
1.如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。
2.如果在观察者和观察目标之间有循环依赖的话,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。
3.观察者模式没有相应的机制让观察者知道所观察的目标对象是怎么发生变化的,而仅仅只是知道观察目标发生了变化。
适用场景:
1.当一个抽象模型有两个方面, 其中一个方面依赖于另一方面。将这二者封装在独立的对象中以使它们可以各自独立地改变和复用。
2.当对一个对象的改变需要同时改变其它对象, 而不知道具体有多少对象有待改变。
3.当一个对象必须通知其它对象,而它又不能假定其它对象是谁。换言之, 你不希望这些对象是紧密耦合的。