/注:场景和例子引用github上的设计模式。传送门:https://github.com/iluwatar/java-design-patterns/tree/master/observer
场景:
一个天气预报系统,凡是订阅了改天气预报的,当天气发生改变的时候就把通知发送给所有订阅该天气预报的人。如兽人族(Orcs)和霍比特人(Hobitts)订阅了该天气预报系统,当天气从晴变成雨天的时候,兽人族和霍比特人就收到了天气变成雨天的通知。 如果还有其他订阅了该天气预报的如人族(Terran),只需要在WeatherObserver中加入该人族(Terran)订阅者,这样当天气发生改变的时候,人族也会收到天气通知。
意图:
定义对象间的一对多关系,当一个对象发生状态更改时,会自动通知并更新所有依赖者。
具体实现:
订阅天气预报系统的即是观察者(如Orcs、Hobitts),把这些订阅天气预报的人抽象成接口(WeatherObserver),所有订阅该天气预报的人都得实现这个接口才行。另外在天气类(Weather)中也得开放一个接口让他们订阅,如使用List<Subscriber> observers 和写一个方法 addObserver(WeatherObserver obs)来让他们订阅,一旦天气发生变化就通知所有订阅该天气的方法 notifyAllObserver()。
talk is cheap,show me the code............................................................................................................................................................................................................................................................(分割线)
先定义天气类型(WeatherType.java),定义为枚举类型(sunny,rainny,windy,clod)
public enum WeatherType { SUNNY,WINDY,CLOD,RAINY; @Override public String toString(){ return this.name().toLowerCase(); } }
再将订阅者抽象成一个接口(WeatherObserver.java),所有订阅天气预报系统的必须实现这个接口。
public interface WeatherObserver { public void Upadate(WeatherType currentWeather); }
天气类(Weather)负责定义数据结构来接收订阅者和通知更新订阅者天气变化。使用List<WeatherObserver>来接收订阅者,定义方法addObserver来接收新的订阅者和notifyAllObsever方法来更新订阅者天气状态。
import java.util.ArrayList; import java.util.List; public class Weather { private WeatherType currentWeather; private List<WeatherObserver> observers; public Weather(){ observers=new ArrayList<>(); currentWeather=WeatherType.SUNNY; } public void addObservers(WeatherObserver observer){ observers.add(observer); } public void removeObservers(WeatherObserver observer){ observers.remove(observer); } public void timePasses(){ WeatherType[] enumValues=WeatherType.values(); /* for(WeatherType type:enumValues){ System.out.println(type); }*/ //改变天气,让天气的enum类型往后推一个 currentWeather=enumValues[(currentWeather.ordinal() + 1) % enumValues.length]; System.out.println("The weather change to "+currentWeather); notifyAllObserver(); } public void notifyAllObserver(){ for(WeatherObserver obs: observers){ obs.Upadate(currentWeather); } } }
添加两个订阅天气预报的“人”,一个兽族(Orcs.java)一个霍比特人(Hobitts.java)。 如上面所说,要订阅该天气预报必须实现WeatherOberver接口。
public class Orcs implements WeatherObserver{ @Override public void Upadate(WeatherType currentWeather) { switch(currentWeather){ case CLOD: System.out.println("The orcs are freezing cold"); break; case SUNNY: System.out.println("The sun hurts the orcs‘ eyes."); break; case RAINY: System.out.println("The orcs are dripping wet."); break; case WINDY: System.out.println("The orc smell almost vanishes in the wind."); default: break; } } public class Hobbits implements WeatherObserver{ @Override public void Upadate(WeatherType currentWeather) { switch(currentWeather){ case CLOD: System.out.println("The Hobbits are freezing cold"); break; case SUNNY: System.out.println("The sun hurts the Hobbits‘ eyes."); break; case RAINY: System.out.println("The Hobbits are dripping wet."); break; case WINDY: System.out.println("The Hobbits smell almost vanishes in the wind."); default: break; } } }
写一个主类来测试一下改功能(App.java)
public class APP { public static void main(String[] args){ Weather weather=new Weather(); weather.addObservers(new Orcs()); weather.addObservers(new Hobbits()); //weather.addObservers(new Terran()); weather.timePasses(); //weather.timePasses(); } }
输出:(即天气从Sunny变成Windy的时候,订阅该天气的Orcs和Hobitts都收到了天气改变的通知)
The weather change to windy The orc smell almost vanishes in the wind. The Hobbits smell almost vanishes in the wind.
在以下任何情况下使用观察者模式:
- 抽象有两个方面,一个依赖于另一个。将这些方面封装在单独的对象中可以让您独立地更改和重用它们
- 当更改一个对象需要更改其他对象时,您不知道需要更改多少个对象
- 当一个对象应该能够通知其他对象,而不需要假设这些对象是谁。换句话说,你不希望这些对象紧密结合