一、什么是观察者模式
观察者模式属于行为型模式。在程序设计中,观察者模式通常由两个对象组成:观察者和被观察者。当被观察者状态发生改变时,它会通知所有的观察者对象,使他们能够及时做出响应,所以也被称作“发布-订阅模式”。
二、特点
三、组成
五、观察者模式实现
下面以报纸和报纸的订阅者为例,假设你在订阅一份报纸,每天早上送到你门口。你订阅的这份报纸就是被观察者。你和其他的订阅者是观察者。当报纸被送到你门口时,它会自动通知所有的订阅者,让他们知道这份报纸已经到了。
5.0 UML类图
5.1 Newspaper
报纸接口,也就是被观察者接口,包含添加、删除、通知三个动作。
/**
*
* 报纸接口,即被观察者接口
*/
public interface Newspaper {
/**
* 添加订阅者
* @param subscriber
*/
void addSubscriber(Subscriber subscriber);
/**
* 移除订阅者
* @param subscriber
*/
void removeSubscriber(Subscriber subscriber);
/**
* 通知订阅者
* @param message
*/
void notifySubscribers(String message);
}
5.2 NewspaperImpl
报纸接口Newspaper的实现类,维护了一个订阅者列表,当报纸到达时会通知所有订阅者。
import java.util.ArrayList;
import java.util.List;
/**
*
* 报纸实现类
*/
public class NewspaperImpl implements Newspaper{
//订阅者集合
List<Subscriber> subscribers = new ArrayList<>();
//添加订阅者
@Override
public void addSubscriber(Subscriber subscriber) {
subscribers.add(subscriber);
}
//移除订阅者
@Override
public void removeSubscriber(Subscriber subscriber) {
subscribers.remove(subscriber);
}
//通知订阅者
@Override
public void notifySubscribers(String message) {
for (Subscriber s : subscribers) {
s.update(message);
}
}
}
5.3 Subscriber
订阅者接口
/**
*
* 订阅者(即观察者)接口
*/
public interface Subscriber {
void update(String message);
}
5.4 SubscriberImpl
订阅者接口的实现类,用于接收报纸到达的通知。
import lombok.AllArgsConstructor;
import lombok.Data;
/**
*
* 具体订阅者
*/
@Data
@AllArgsConstructor
public class SubscriberImpl implements Subscriber{
private String name;
@Override
public void update(String message) {
System.out.println(name + "---接到消息: " + message);
}
}
5.5 测试
测试代码中,创建了一个NewspaperImpl对象,然后注册了两个订阅者(李老头、王奶奶),执行了两次通知,第一次通知,李老头和王奶奶都收到了消息,第二次通知前,移除了李老头这个订阅者,只有王奶奶收到了通知。
@SpringBootTest
public class TestObserver {
@Test
void testObserver(){
Newspaper newspaper = new NewspaperImpl();
Subscriber li = new SubscriberImpl("李老头");
Subscriber wang = new SubscriberImpl("王奶奶");
//李老头和王奶奶订阅了报纸
newspaper.addSubscriber(li);
newspaper.addSubscriber(wang);
//报纸到了,通知订阅者
newspaper.notifySubscribers("今天的报纸到了!!!");
//李老头取消订阅了,移除
newspaper.removeSubscriber(li);
newspaper.notifySubscribers("明天的报纸还是这个点到!!!");
}
}