14. 观察者模式
举例,当 excel 表格中的数据发生改变时,基于该数据绘制的图片,也同步发生改变。
自动更新,不需要手动更新。
观察者模式应用比较广泛,又被称为“发布-订阅”模式。
它用来定义对象间一种一对多的依赖关系,当一个对象的状态发生变化时,所有依赖它的对象都得到通知并被自动更新。
观察者模式的角色有:抽象主题、具体主题(发布者)、抽象观察者和具体观察者(订阅者)。
使用场景:当一个抽象模型有两个方面,其中一个方面依赖另一个方面。
- 将这两者封装在独立对象中以使它们可以各自独立地改变和复用;
- 当对一个对象的改变需要同时改变其它对象,而不知道具体有多少对象待改变;
- 当一个对象必须通知其它对象,而它又不能假定其它对象是谁。
换言之,你不希望这些对象是紧耦合的。
优点:目标和观察者之间的抽象耦合最小;支持广播通信。
from abc import ABCMeta, abstractmethod
# 定义高层代码和低层代码之间的约束关系
# 定义两个接口:1)观察者(订阅者)、2)
from abc import ABCMeta, abstractmethod
# 抽象的订阅者
class Observer(metaclass=ABCMeta):
@abstractmethod # (接口)
def update(self, notice): # notice 是一个 Notice 类的对象
"""
:param notice: Notice类的对象
:return:
"""
pass
# 抽象的发布者:可以是接口,子类不需要实现,所以不需要定义抽象方法!
class Notice:
def __init__(self):
self.observers = [] # 该列表,用来存储所有观察者,即用来储存该发布者(observers)的所有订阅者类型
def attach(self, obs): # 绑定(订阅)
self.observers.append(obs)
def detach(self, obs): # 解绑(取消订阅)
self.observers.remove(obs)
def notify(self):
"""
推送(发布新的消息后,通知观察者对象),一旦发布者发生变化的时候,它要通知订阅者,
:return:
"""
for obs in self.observers: # 对于 self.observers里的每一个观察者对象,调用其update方法来更新
obs.update(self) # self 表示 Notice 对象的本身
# 具体的发布者
class StaffNotice(Notice): # StaffNotice继承了Notice类,那它就是一个发布者
def __init__(self, company_info): # company_info 可以理解为一个叫 公告 的属性
super().__init__() # 调用父类对象声明observers属性
self.__company_info = company_info # 创建company_info属性,'__'双下划线表示私有
@property
def company_info(self): # 用来读私有属性 __company_info
return self.__company_info
@company_info.setter
def company_info(self, info): # 用来修改私有属性 __company_info
self.__company_info = info
self.notify()
# 具体的订阅者
class Staff(Observer): # 从 Observer 接口里继承,所以下面要实现 update 方法
def __init__(self):
self.company_info = None # 定义了自己的消息列表
def update(self, notice): # 这个 notice 对象其实就是 StaffNotice 对象
self.company_info = notice.company_info
staff_notice = StaffNotice('初始化公司信息')
staff1 = Staff()
staff2 = Staff()
staff_notice.attach(staff1)
staff_notice.attach(staff2)
# print(staff1.company_info) None # 输出私有属性 __company_info
# print(staff2.company_info) None
staff_notice.company_info = '假期放假通知!'
print(staff1.company_info)
print(staff2.company_info)
staff_notice.detach(staff2)
staff_notice.company_info = '明天开会!'
print(staff1.company_info)
print(staff2.company_info)
"""
假期放假通知!
假期放假通知!
明天开会!
假期放假通知!
"""
注意,并没有把 Notice 类定义成接口,原因是,这些方法都不用到子类去实现:
参考资料
[1] Python 之常用设计模式 2020.3;