Excel 中的数据与折线图、饼状图、柱状图之间的关系;MVC 模式中的模型与视图的关系;事件模型中的事件源与事件处理者。所有这些,如果用观察者模式来实现就非常方便。
模式的定义与特点
观察者(Observer)模式的定义:指多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。这种模式有时又称作发布-订阅模式、模型-视图模式,它是对象行为型模式。
观察者模式是一种对象行为型模式,其主要优点如下
- 降低了目标与观察者之间的耦合关系,两者之间是抽象耦合关系。符合依赖倒置原则。
- 目标与观察者之间建立了一套触发机制。
它的主要缺点如下
- 目标与观察者之间的依赖关系并没有完全解除,而且有可能出现循环引用。
- 当观察者对象很多时,通知的发布会花费很多时间,影响程序的效率。
个人理解
与桥接非常相近,继承关系上一致,区别在于桥接的具体对象与抽象对象一对一,在抽象对象下的扩展对象内直接执行抽象对象下的方法;观察者则为一对多的关系,在抽象目标内定义的是抽象观察者组。模式的结构与实现
实现观察者模式时要注意具体目标对象和具体观察者对象之间不能直接调用,否则将使两者之间紧密耦合起来,这违反了面向对象的设计原则。
进口公司和出口公司都会观察汇率变化并作出反应。(汇率变化后执遍历行所有公司的response反应方法)
使用时先创建观察者(进口公司)和具体目标(汇率-人民币汇率),汇率对象add添加所有观察它的对象,在自身汇率变化后遍历执行公司的反映方法。 ```csharp //抽象观察者 public interface Observer { void response(); } //具体观察者 public class ConcreteObserverA : Observer { public void response() {
} } //抽象目标 public abstract class SubjectTarget { protected ListDebug.Log("具体观察者A做出反应");
observers = new List (); public void add(Observer observer) {
} public void remove(Observer observer) {observers.Add(observer);
} public abstract void notifyObserver(); } //具体目标 public class ConcreteSubjectTarget : SubjectTarget { public override void notifyObserver() {observers.Remove(observer);
} }Debug.Log("具体目标发生改变");
foreach (var obs in observers)
{
obs.response();
}
```csharp
void Start()
{
Observer observer = new ConcreteObserverA();
SubjectTarget target = new ConcreteSubjectTarget();
target.add(observer);
target.notifyObserver();
}
//执行结果
//具体目标发生改变
//具体观察者A做出反应
模式的应用场景
通过前面的分析与应用实例可知观察者模式适合以下几种情形
- 对象间存在一对多关系,一个对象的状态发生改变会影响其他对象。
- 当一个抽象模型有两个方面,其中一个方面依赖于另一方面时,可将这二者封装在独立的对象中以使它们可以各自独立地改变和复用。
- 实现类似广播机制的功能,不需要知道具体收听者,只需分发广播,系统中感兴趣的对象会自动接收该广播。
- 多层级嵌套使用,形成一种链式触发机制,使得事件具备跨域(跨越两种观察者类型)通知。