模式说明
当一个对象变化时,所有与他有关联关系的对象都将因为这个变化而变化,将前者称为目标对象,后者称为观察者对象。例如以人行道上的红绿灯为目标对象,行人和车辆为观察者对象。当红绿灯从红灯变为绿灯时,行人因为这个变化而从等待转变为过马路,而汽车则从通过转变为等待,这些观察者都对目标的变化做出了反映。此类场景可以用观察者模式实现。
本示例以人民币汇率为目标对象,进口公司和出口公司作为观察者对象。演示目标对象变化时,如何通知所有观察者做出其响应的动作。
结构
抽象目标类
观察者需要观察的对象,类内实现添加和删除观察者的普通方法,定义一个通知所有观察者的抽象方法。
具体目标类
继承抽象目标类,实现抽象方法。
抽象观察者类
持有一个name属性,并有对应的getter/setter。定义了一个响应目标对象变化的抽象方法action,被目标类通知时执行。
具体观察者类
继承抽象观察者类,实现抽象方法action。
代码演示
package com.yukiyama.pattern.behavior;import java.util.ArrayList;import java.util.List;/*** 观察者模式*/public class ObserverDemo {public static void main(String[] args) {// 声明一个具体目标对象,人民币汇率Rate rate = new RMBRate();// 声明两个不同的观察者对象,进口公司和出口公司Company watcher1 = new ImportCompany();Company watcher2 = new ExportCompany();// 将观察者都加入到目标对象中,以便目标能够在自身产生变化时通知所有观察者rate.add(watcher1);rate.add(watcher2);// 当汇率升高3.5%时,进出口公司分别对这个变化做出应对rate.change(0.035);// 当汇率下降6.8%时,进出口公司分别对这个变化做出应对rate.change(-0.068);}}/*** 抽象目标类(被观察事物)* 持有一个观察者类实例的集合,实现两个非抽象方法,add方法用来添加观察者* 实例,remove方法用来删除观察者实例。另有一个抽象方法change,传入变化。*/abstract class Rate{protected List<Company> companies = new ArrayList<>();public void add(Company company) {companies.add(company);}public void remove(Company company) {companies.remove(company);}public abstract void change(double changedRate);}/*** 具体目标类* 继承抽象目标类,实现抽象方法change,在方法内遍历其持有的所有观察者,* 传入变化,执行观察者的action方法。*/class RMBRate extends Rate{@Overridepublic void change(double changedRate) {System.out.printf("人民币汇率变动%.1f%%\n", changedRate*100);for(Company company : this.companies) {company.action(changedRate);}}}/*** 抽象观察者类* 持有一个name属性,并有对应的getter/setter。定义了一个action抽象方法。* 下例以公司为抽象观察者类。*/abstract class Company{private String name;public String getName() {return name;}public void setName(String name) {this.name = name;}public abstract void action(double changedRate);}/*** 具体观察者类* 通过无参构造器初始化name。实现抽象观察者类中的抽象方法action。* 下例是进口公司观察者,action内实现当人民币汇率降低时减少相应比例* 的进口量,升高时增加相应比例的进口量。*/class ImportCompany extends Company{public ImportCompany() {this.setName("进口公司");}@Overridepublic void action(double changedRate) {System.out.printf("%s动作:\n", this.getName());if(changedRate < 0) {System.out.printf("人民币汇率下降了%.1f%%,减少%.1f%%进口量。\n",-changedRate*100, -changedRate*100);} else {System.out.printf("人民币汇率上升了%.1f%%,增加%.1f%%进口量。\n",changedRate*100, changedRate*100);}}}/*** 具体观察者类* 下例是出口公司观察者,action内实现当人民币汇率降低时增加相应比例* 的进口量,升高时减少相应比例的进口量。*/class ExportCompany extends Company{public ExportCompany() {this.setName("出口公司");}@Overridepublic void action(double changedRate) {System.out.printf("%s动作:\n", this.getName());if(changedRate < 0) {System.out.printf("人民币汇率下降了%.1f%%,增加%.1f%%出口量。\n",-changedRate*100, -changedRate*100);} else {System.out.printf("人民币汇率上升了%.1f%%,下降%.1f%%出口量。\n",changedRate*100, changedRate*100);}}}
