观察者模式,以前看过,今天想了一下,似乎有点模糊了
行为模式,为实现此模式,应该具备这几个角色:被观察的主题,观察者。这是必须的!一般来说,为使构建的东西有一定的可用性,会进行进一步的分工。主题角色再分为抽象主题和具体主题,同样滴划分观察者角色为抽象观察者及具体观察者角色。这就是将在此文中提及的 Subject, ConcreteSubject, Observer, ConcreteObserver。
按照《JAVA 与模式》书来的,实现了最简单的两个版本。
版本一:
/** 抽象主题角色,属于被观察者 */
public interface Subject {
/**
* 登记一个新的观察者对象
*/
public void attach(Observer observer);
/**
* 删除一个观察者对象
*/
public void detach(Observer observer);
/**
* 通知所有登记过的观察者对象
*/
void notifyObservers();
}
/** 具体主题角色,属于被观察者 */
public class ConcreteSubject implements Subject {
private Vector observersVector = new Vector();
/** 这里省略了几个方法的实现,无非是往向量中添加 Observer, 没什么好写的 */
/**
* 通知所有登记过的观察者对象
* 这里和书有一点点的不一样,我用的是 Iterator 接口来实现
* 书上的是 Enumeration ,但大同小异。
*/
public void notifyObservers() {
Iterator it = observersVector.iterator();
while(it.hasNext()) {
((Observer)it.next()).update();
}
}
}
/** 抽象观察者角色 */
public interface Observer {
/**
* 调用这个方法会更新自己
*/
void update();
}
/** 具体观察者角色 */
public class ConcreteObserver implements Observer {
/**
* 调用这个方法会更新自己
*/
public void update() {
System.out.println(this + " I am notified");
}
}
客户端的 main() 如下:
public static void main(String[] args) {
Subject subject = new ConcreteSubject();
Observer observer1 = new ConcreteObserver();
Observer observer2 = new ConcreteObserver();
subject.attach(observer1);
subject.attach(observer2);
subject.notifyObservers();
}
版本一的实现基本上与书上的没有什么不同,而版本二则有一点点的不一样。版本二只改变了版本一中的两个被观察者的一些细节。在此贴上:
/**
* 这是一个重量级的抽象主题角色,将子类的一些方法移至此
* 增加了抽象类的重量而减少了具体主题类的分量
*/
public abstract class Subject {
/**
* 保存所有对观察者对象的引用
*/
private Vector observersVector = new Vector();
/**
* 更改主题的状态
* 在书上没有该方法,为了在客户端能够用基类声明对象,将该方法的声明移至此
*/
public abstract void change(String newState);
/** 在这里也把往主题角色中添加/删除观察者对象的方法省略了 */
/**
* 通知所有登记过的观察者对象
*/
public void notifyObservers() {
Iterator it = observersVector.iterator();
while(it.hasNext()) {
((Observer)it.next()).update();
}
}
}
/**
* 轻量级的具体主题类角色
*/
public class ConcreteSubject extends Subject {
private String state;
/**
* 更改主题的状态
*/
public void change(String newState) {
state = newState;
this.notifyObservers();
}
}
客户端的代码是完全一样的。
其实这样一个模式是相对比较简单的,当然,简单并不意味着力量不大,呵呵。用另外一个形式来表示:
public class 被观察者 {
public void 通知观察者() {
观察者对象.有情况发生();
}
}
public class 观察者 {
public void 有情况发生() {
/* 在这里做你想做的事情 */
}
}
当再把这两个类分解成抽象的和具体的后,一个相对完整的观察者模式的实现就出来了。