欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 科技 > 名人名企 > 基于Python学习《Head First设计模式》第二章 观察者模式

基于Python学习《Head First设计模式》第二章 观察者模式

2025/10/1 0:01:04 来源:https://blog.csdn.net/wzx77/article/details/148364440  浏览:    关键词:基于Python学习《Head First设计模式》第二章 观察者模式

观察者模式

定义

在这里插入图片描述

类图

在这里插入图片描述

应用项目:气象站

要求

在这里插入图片描述
在这里插入图片描述

实现类图

在这里插入图片描述

实现方式1-更新所有观察者的所有数据

搭建观察者模式框架

在这里插入图片描述


class Observer:"""观察者"""@abstractmethoddef update(self,temp, humidity, pressure):"""由主题调用更新"""passclass Subject:"""主题"""@abstractmethoddef register_observer(self, observer: Observer):"""注册"""pass@abstractmethoddef remove_observer(self, observer: Observer):"""移除"""pass@abstractmethoddef notify_observers(self):"""通知所有观察者"""passclass DisplayElement:def __init__(self):pass@abstractmethoddef display(self):"""显示"""pass

实现WeatherData

在这里插入图片描述

class WeatherData(Subject):def __init__(self):self._observers: list[Observer] = []self._temp: Optional[float] = Noneself._pressure: Optional[float] = Noneself._humidity: Optional[float] = Nonedef register_observer(self, observer: Observer):self._observers.append(observer)def remove_observer(self, observer: Observer):self._observers.remove(observer)def notify_observers(self):for obs in self._observers:obs.update(self._temp, self._humidity, self._pressure)def measurements_changed(self):self.notify_observers()def set_measurements(self, temp: float, humidity: float, pressure: float):self._temp = tempself._humidity = humidityself._pressure = pressureself.measurements_changed()

实现布告板

在这里插入图片描述

class CurrentConditionsDispaly(Observer, DisplayElement):def __init__(self):self._weather_data = WeatherData()self._weather_data.register_observer(self)self._temp: Optional[float] = Noneself._pressure: Optional[float] = Noneself._humidity: Optional[float] = Nonedef update(self, temp, humidity, pressure):self._temp = tempself._humidity = humidityself._pressure = pressureself.display()def display(self):print(id(self))  # 打印当前示例id,用于区分print(f"温度:{self._temp}"f"湿度:{self._humidity}"f"气压:{self._pressure}")

测试&运行

在这里插入图片描述

if __name__ == '__main__':weather = WeatherData()display = CurrentConditionsDispaly(weather)weather.set_measurements(80, 80, 2.34)display2 = CurrentConditionsDispaly(weather)weather.set_measurements(90, 70.5, 5)

完整代码


from abc import abstractmethod
from typing import Optionalclass Observer:"""观察者"""@abstractmethoddef update(self, temp, humidity, pressure):"""由主题调用更新"""passclass Subject:"""主题"""@abstractmethoddef register_observer(self, observer: Observer):"""注册"""pass@abstractmethoddef notify_observers(self):"""通知所有观察者"""pass@abstractmethoddef remove_observer(self, observer: Observer):"""移除"""passclass DisplayElement:@abstractmethoddef display(self):"""显示"""passclass WeatherData(Subject):def __init__(self):self._observers: list[Observer] = []self._temp: Optional[float] = Noneself._pressure: Optional[float] = Noneself._humidity: Optional[float] = Nonedef register_observer(self, observer: Observer):self._observers.append(observer)def remove_observer(self, observer: Observer):self._observers.remove(observer)def notify_observers(self):for obs in self._observers:obs.update(self._temp, self._humidity, self._pressure)def measurements_changed(self):self.notify_observers()def set_measurements(self, temp: float, humidity: float, pressure: float):self._temp = tempself._humidity = humidityself._pressure = pressureself.measurements_changed()class CurrentConditionsDispaly(Observer, DisplayElement):def __init__(self, weather_data: WeatherData):self._weather_data = weather_dataself._weather_data.register_observer(self)self._temp: Optional[float] = Noneself._pressure: Optional[float] = Noneself._humidity: Optional[float] = Nonedef update(self, temp, humidity, pressure):self._temp = tempself._humidity = humidityself._pressure = pressureself.display()def display(self):print(id(self))print(f"温度:{self._temp}\n"f"湿度:{self._humidity}\n"f"气压:{self._pressure}\n")if __name__ == '__main__':weather = WeatherData()display = CurrentConditionsDispaly(weather)weather.set_measurements(80, 80, 2.34)display2 = CurrentConditionsDispaly(weather)weather.set_measurements(90, 70.5, 5)"""运行结果:
4481847152
温度:80
湿度:80
气压:2.344481847152
温度:90
湿度:70.5
气压:54481845424
温度:90
湿度:70.5
气压:5
"""

实现方式2-通知观察者自行拉取数据更新

类图

在这里插入图片描述

代码实现

在这里插入图片描述


from abc import abstractmethod, ABC
from typing import Optional, Unionclass Observer(ABC):"""观察者"""@abstractmethoddef update(self, observerable, arg):"""由主题调用更新"""passclass Observerable:"""可观察者"""_observers = []changed = Falsedef register_observer(self, observer: Observer):self._observers.append(observer)def remove_observer(self, observer: Observer):self._observers.remove(observer)def notify_observers(self, arg=None):if self.changed:for obs in self._observers:obs.update(self, arg)else:self.changed = Falsedef set_changed(self):self.changed = Trueclass DisplayElement:@abstractmethoddef display(self):"""显示"""passclass WeatherData(Observerable):_temp: float  # python新版本可不指定默认值_humidity: float_pressure: floatdef measurements_changed(self):self.set_changed()self.notify_observers()def get_temperature(self) -> float:return self._tempdef get_humility(self) -> float:return self._humiditydef get_pressure(self) -> float:return self._pressuredef set_measurements(self, temp: float, humidity: float, pressure: float):self._temp = tempself._humidity = humidityself._pressure = pressureself.measurements_changed()class CurrentConditionsDispaly(Observer, DisplayElement):_temp: float  # python新版本可不指定默认值_humidity: Optional[float]  # 值类型可以为float和None时,可以用 Optional[float] 相当于 Union[float, None]_pressure: Union[float, None]def __init__(self, observerable: Observerable):self._observerable = observerableobserverable.register_observer(self)def update(self, observerable, arg):if isinstance(observerable, WeatherData):self._temp = observerable.get_temperature()self._pressure = observerable.get_pressure()self._humidity = observerable.get_humility()self.display()def display(self):print(id(self))print(f"温度:{self._temp}\n"f"湿度:{self._humidity}\n"f"气压:{self._pressure}\n")if __name__ == '__main__':weather = WeatherData()display = CurrentConditionsDispaly(weather)weather.set_measurements(80, 80, 2.34)display2 = CurrentConditionsDispaly(weather)weather.set_measurements(90, 70.5, 5)

总结

在这里插入图片描述

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词