[设计模式] Observer 观察者模式

本文深入讲解观察者模式的概念与应用,通过气象站示例详细解释主题与观察者之间的交互方式,探讨模式的设计原则,并提供了完整的Java实现代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Observer 观察者模式

观察者模式定义了对象之间一对多依赖,当一个对象改变状态时,它所有的依赖者都会受到通知并且更新

简单地说:你订阅公众号,公众号为主题,你是观察者,公众号有了新动态你立马收到通知,取消订阅后就不会收到通知了

设计原则

  • 找出程序中会变化的方面,然后将其和固定不变的方面分离
  • 针对接口编程,而不是针对实现编程
  • 多用组合,少用继承

1. 示例结构图

以HeadFirst书中的气象站为例类图如下(代码在最后):

气象站,发布气象数据,为主题

WeatherData 

布告栏,显示气象数据,为观察者

CurrentConditionsDisplay
ForecastDisplay
StatisticsDisplay 

2. 小结

  • JDK中观察者通过一个类实现,限制了他的复用能力Java 9后废弃
  • 观察者和被观察者是抽象耦合的

3. 示例代码

/**
*主题接口
*声明了添加、删除、通知观察者的方法
*
*/
public interface Subject {
	public void registerObserver(Observer o);
	public void removeObserver(Observer o);
	public void notifyObserver();
}

/**
*观察者接口
*声明了update()方法,当主题调用notifyObserver()方法时,观察者的update()方法被调用
*
*/
public interface Observer {
	public void update(float temp,float humidity,float pressure);
}
/**
*显示接口
*与观察者设计模式无关,用于为观察者提供显示数据的接口
*
*/
public interface DisplayElement {
	public void display();
}

接口的实现类:

import java.util.ArrayList;
/**
*主题的实现类
*
*/
public class WeatherData implements Subject{
	private float temperature;
	private float humidity;
	private float pressure;
    //ArrayList用于存放订阅了该主题的观察者,面向接口编程
	private ArrayList<Observer> observers;
	
	public WeatherData() {
		observers = new ArrayList<Observer>();
	}
	
	public void registerObserver(Observer o) {
		observers.add(o);
	}

	public void removeObserver(Observer o) {
		int i = observers.indexOf(o);
		if(i >= 0) {
			observers.remove(i);
		}
	}

	public void notifyObserver() {
		for(int i = 0; i<observers.size();i++) {
            //这里调用了观察者的update()方法,多态
			observers.get(i).update(temperature, humidity, pressure);
		}
	}
	
	public void measurementsChanged() {
		notifyObserver();
	}
	//这个函数用于模拟气象站接收到了新的气象数据
	public void setMeasurements(float temp,float humidity,float pressure) {
		this.temperature = temp;
		this.humidity = humidity;
		this.pressure = pressure;
		measurementsChanged();
	}
	
}
/**
*观察者的实现类
*实现了Observer接口
*
*/
public class CurrentConditionsDisplay implements Observer,DisplayElement{
	
	private float temperature;
	private float humidity;
	private Subject subject;
	//在构造函数中将自己添加到主题中
	public CurrentConditionsDisplay(Subject s) {
		this.subject = s;
		this.subject.registerObserver(this);
	}
	
	public void display() {
		System.out.println("Current conditions: " + temperature + "F degrees and " + humidity + "% humidity");
	}

	public void update(float temp, float humidity, float pressure) {
		this.temperature = temp;
		this.humidity = humidity;
		display();
	}

}

测试类:

public class WeathetStation {
	public static void main(String[] args) {
		WeatherData weatherData = new WeatherData();
		
		CurrentConditionsDisplay currentDisplay = new CurrentConditionsDisplay(weatherData);
		
		weatherData.setMeasurements(80, 65, 30.4f);
		weatherData.setMeasurements(85, 60, 31.1f);
		weatherData.setMeasurements(75, 53, 28.5f);
		
	}
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值