观察者模式也叫发布-订阅模式,其定义如下:定义对象间一种一对多的依赖关系,使得当该对象状态改变时,所有依赖于它的对象都会得到通知,并被自动更新。
观察者模式的通知方式可以通过直接调用等同步方式实现(如函数调用,HTTP接口调用等),也可以通过消息队列异步调用(同步调用指被观察者发布消息后,必须等所有观察者响应结束后才可以进行接下来的操作;异步调用指被观察者发布消息后,即可进行接下来的操作。)
许多开源的消息队列就直接支持发布-订阅模式
优点:
1、观察者与被观察者之间是抽象耦合的;
2、可以将许多符合单一职责原则的模块进行触发,也可以很方便地实现广播。
应用场景:
1、消息交换场景。如上述说到的消息队列等;
2、多级触发场景。比如支持中断模式的场景中,一个中断即会引发一连串反应,就可以使用观察者模式。
缺点
1、观察者模式可能会带来整体系统效率的浪费;
2、如果被观察者之间有依赖关系,其逻辑关系的梳理需要费些心思。
流程图:
Python代码实现:
"""
观察者模式
从业务流程的实现角度,实现该火警报警器。
"""
# class AlarmSensor:
# def run(self):
# print ("报警器...")
#
#
# class WaterSprinker:
# def run(self):
# print ("洒水器...")
#
#
# class EmergencyDialer:
# def run(self):
# print ("拨 119...")
# 将三个类提取共性,泛化出“观察者”类,并构造被观察者。
class Observer:
def update(self):
pass
class AlarmSensor (Observer):
def update(self, action):
print ("收到报警: %s" % action)
self.runAlarm ()
def runAlarm(self):
print ("报警器...")
class WaterSprinker (Observer):
def update(self, action):
print ("收到洒水: %s" % action)
self.runSprinker ()
def runSprinker(self):
print ("洒水器...")
class EmergencyDialer (Observer):
def update(self, action):
print ("收到拨号: %s" % action)
self.runDialer ()
def runDialer(self):
print ("拨 119...")
# 被观察者
class Observed:
observers=[]
action=""
def addObserver(self,observer):
self.observers.append(observer)
def notifyAll(self):
for obs in self.observers:
obs.update(self.action)
class smokeSensor(Observed):
def setAction(self,action):
self.action=action
def isFire(self):
return True
if __name__=="__main__":
alarm=AlarmSensor()
sprinker=WaterSprinker()
dialer=EmergencyDialer()
smoke_sensor=smokeSensor()
smoke_sensor.addObserver(alarm)
smoke_sensor.addObserver(sprinker)
smoke_sensor.addObserver(dialer)
if smoke_sensor.isFire():
smoke_sensor.setAction("着火!")
smoke_sensor.notifyAll()
原文:https://yq.aliyun.com/articles/71066?spm=a2c4e.11153940.blogcont280715.27.175692aawkXRdB