一,定义。
观察者模式:定义了一个一对多的依赖关系,让一个或多个观察者对象监听一个主题对象。这样一来,当被观察者状态发生改变时,需要通知响应的观察者,使这些观察者对象能够自动更新。
二,关键要素。
1,主题:是观察者观察的对象,一个主题必须具备下面三个特征——持有监听的观察者的引用+支持增加和删除观察者+主题状态改变,通知观察者。
2,观察者:当主题发生变化,收到通知进行具体的处理是观察者必须具备的特征。
三,为什么要用这种模式。
这里举一个例子来说明,牛奶送奶站就是主题,订奶客户为监听者,客户从送奶站订阅牛奶后,会每天收到牛奶。如果客户不想订阅了,可以取消,以后就不会收到牛奶。
1,松耦合:观察者增加或删除无需修改主题的代码,只需调用主题对应的增加或者删除的方法即可
+主题只负责通知观察者,但无需了解观察者如何处理通知。举个例子,送奶站只负责送递牛奶,不关心客户是喝掉还是洗脸
+观察者只需等待主题通知,无需观察主题相关的细节。还是那个例子,客户只需关心送奶站送到牛奶,不关心牛奶由哪个快递人员,使用何种交通工具送达。
2,通知不错过:由于被动接受,正常情况下不会错过主题的改变通知。而主动获取的话,由于时机选取问题,可能导致错过某些状态。
四,Java实现。
Java中有观察者模式使用的API:
java.util.Observable——这是一个类,而非接口,主题需要继承这个类。
import java.util.Observable;
public class MilkProvider extends java.util.Observable{
public void action(Object arg) {
super.setChanged();// setChanged 是protected方法,必须由本来来调用。。。
super.notifyObservers(arg);
}
}
java.util.Observer——这是一个接口,监听者需要实现这个接口。
import java.util.Observer;
/**
* 监听者, 此update方法自动由Observer类调用,
*
* 而 update方法主要是调用 业务方法, 当然,我们也可以在这个方法中直接的业务逻辑处理, 而不用调来调去
*
* 需要继承Observer是因为 它是和Observable配套使用的
*/
public class Consumer implements Observer {
@Override
public void update(Observable arg0, Object arg1) {
System.out.println("Consumer update..." + arg0 + ";arg1=" + arg1);
}
}
测试类——
import java.util.Observer;
public class Test {
public static void main(String[] args) {
Observer consumer = new Consumer();
MilkProvider provider = new MilkProvider();
provider.addObserver(consumer);
provider.action("Hello,World!");
}
}
上述代码完成了——将consumer加入到主题provider的观察者行列+provider设置状态变化,通知持有的观察者+观察者consumer收到通知,打印日志处理。
PS:未完,持续更新。。。