观察者模式
其目的是用来定义对象间的一种一对多的依赖关系,以便当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。观察者塔式统通常被用来实现事件处理系统。
观察者模式不仅被广泛应用于软件界面元素之间的交互,在业务对象之间的交互、权限管理等方面也有广泛的应用。
在观察者模式中,一个目标物件管理所有相依于它的观察者物件,并且在它本身的状态改变时主动发出通知。通常通过调用各观察者所提供的方法来实现。此种模式通常被用来实现事件处理系统。
观察者模式完美的将观察者和被观察对象分离开。比如,用户界面可以作为一个观察者,业务数据是被观察者,用户界面观察业务数据的变化,发现数据变化后,就显示在界面上。观察者模式模式必须包含观察者和被观察对象两个角色。
观察者设计模式**优点**:
- 观察者和被观察者是松偶合(抽象耦合)的。改变主题或观察者中的一方,另一方不会受到影响。
- 建立一套触发机制。
观察者设计模式**缺点**:
- 如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。
- 如果在观察者和被观察对象之间有循环依赖的话,被观察对象会触发它们之间进行循环调用,可能导致系统崩溃。
- 观察者模式没有相应的机制让观察者知道所观察的目标对象是怎么发生变化的,而仅仅只是知道观察目标发生了变化。
注意事项:
- JDK中也有自带的观察者模式(Java中已经标为过时,不再建议使用了),但是被观察者是一个类而不是接口,限制了它的复用能力
- 避免循环引用
- 如果顺序执行,某一观察者错误会导致系统卡壳,一般采用异步方式
观察者设计模式包含四个角色:
- 抽象被观察者角色(Subject):也就是一个抽象主题,它把所有对观察者对象的引用保存在一个集合中,每个主题都可以有任意数量的观察者。抽象主题提供一个接口,可以增加和删除观察者角色。一般用一个抽象类和接口来实现。
- 抽象观察者角色(Observer):为所有的具体观察者定义一个接口,在得到主题通知时更新自己。
- 具体被观察者角色(ConcreteSubject):也就是一个具体的主题,在集体主题的内部状态改变时,所有登记过的观察者发出通知。
- 具体观察者角色(ConcreteObserver):实现抽象观察者角色所需要的更新接口,一边使本身的状态与制图的状态相协调。
示例:
假设有三个人,小美(女,22),小王和小李。小美很漂亮,小王和小李是两个程序猿,时刻关注着小美的一举一动。有一天,小美说了一句:“谁来陪我打游戏啊。”这句话被小王和小李听到了,结果乐坏了,蹭蹭蹭,没一会儿,小王就冲到小美家门口了,在这里,小美是被观察者,小王和小李是观察者,被观察者发出一条信息,然后观察者们进行相应的处理,看代码:
public interface Person {
void getMessage(String str);
}
这个接口相当于小王和小李的电话号码,小美发送通知的时候就会拨打getMessage这个电话,拨打电话就是调用接口,看不懂没关系,先往下看
public class LaoWang implements Person {
private String name = "小王";
@Override
public void getMessage(String str) {
System.out.println(name + "接到小美电话:" + str);
}
}
public class LaoLi implements Person {
private String name = "老李";
@Override
public void getMessage(String str) {
System.out.println(name + "接到小美电话:" + str);
}
}
代码很简单,我们再看看小美的代码:
public class XiaoMei {
List<Person> list = new LinkedList<>();
public void addPerson(Person person) {
list.add(person);
}
public void sendMessage() {
for (Person p : list) {
p.getMessage("谁来陪我打游戏,先到先得!");
}
}
public static void main(String[] args) {
LaoLi laoLi = new LaoLi();
LaoWang laoWang = new LaoWang();
XiaoMei xiaoMei = new XiaoMei();
xiaoMei.addPerson(laoLi);
xiaoMei.addPerson(laoWang);
xiaoMei.send
+Message();
}
}