介绍
观察者模式类似于mvc模式,将数据的操作和视图的操作分离开来,我们开发过程中使用的消息队列就是观察者模式,多个消费者作为观察者,盯着是否有自己感兴趣的数据,而生产者作为数据的发布者,将数据推向队列,也间接推向了消费者,从而使每个观察者拿到最新的数据。
区别于MVC模式,MVC模式是数据变动后,不会向视图层推送,也就是说视图层如果想拿到最新的数据,必须重新获取一次。举个例子,两个页面公用一个数据库,从一个页面改动数据后,如果另一个页面要展示最新的数据,就只能刷新一次了。而观察者模式,不需要刷新,当数据变动后,它会主动把数据推送给页面。
举例
我们用代码演示一下,椰奶工厂上了一个大屏,大屏上有两个面板,折线图面板和饼图面板,当椰奶数据变更时,这两个面板也实时同步。
# 数据发布者
class DataPublish:
def __init__(self):
self.observers = []
def add(self, observer):
if observer not in self.observers:
self.observers.append(observer)
else:
print(f"failed to add: {observer}")
def remove(self, observer):
try:
self.observers.remove(observer)
except ValueError as e:
print(f"Failed to remove: {observer}")
def notify(self):
[part.notify(self) for part in self.observers]
class DefaultFormatter(DataPublish):
def __init__(self, name):
DataPublish.__init__(self)
self.name = name
self._data = 0
def __str__(self):
return f"{type(self).__name__}: '{self.name}' has data = {self._data} "
@property
def data(self):
return self._data
@data.setter
def data(self, new_value):
try:
self._data = int(new_value)
self.notify()
except ValueError as e:
print(f"Error: ", {e})
# 观察者
class LineGraph:
def notify(self, publisher):
print(f"我是线图,现在的椰奶数量是{publisher.data}")
# 观察者
class PieGraph:
def notify(self, publisher):
print(f"我是饼图,现在的椰奶数量是{publisher.data}")
if __name__ == '__main__':
dp = DefaultFormatter("椰奶数据发布者")
dp.add(LineGraph())
dp.add(PieGraph())
# 变更数据
print("==== 继续变更数据 ====")
dp.data = 11
print("==== 继续变更数据 ====")
# 变更数据
dp.data = 21
执行结果