13.观察者模式

1.定义

观察者模式应用比较广泛,又被称为“发布-订阅”模式。

它用来定义对象间一种一对多的依赖关系,当一个对象的状态发生变化时,所有依赖它的对象都得到通知并被自动更新。

观察者模式的角色有:

抽象主题

具体主题(发布者)

抽象观察者

具体观察者(订阅者)

一个主题存在一个或多个订阅者,一个订阅者可订阅多个主题,类似于微信公众号的订阅。

具体主题在发布消息时,还要把消息推送到具体的订阅者处。

订阅者具有订阅某个主题消息(绑定 attach)、不订阅某个消息(不绑定 detach)方法。

2.实现

from abc import ABCMeta, abstractmethod

# 抽象的订阅者
class Observer(metaclass=ABCMeta):
    @abstractmethod
    def update(self, notice):
        """
        用于更新订阅者的消息
        :param notice: Notice类的对象
        :return:
        """
        pass

# 抽象的发布者:可以是接口,子类不需要实现,所以不需要定义抽象方法!
class Notice:
    def __init__(self):
        self.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:
            obs.update(self)

# 具体的发布者
class StaffNotice(Notice):
    def __init__(self, company_info):
        super().__init__()  # 调用父类对象声明observers属性
        self.__company_info = company_info

    @property
    def company_info(self):
        return self.__company_info

    @company_info.setter
    def company_info(self, info):
        self.__company_info = info
        self.notify()  # 推送更新订阅者的消息

# 具体的订阅者
class Staff(Observer):
    def __init__(self):
        self.company_info = None

    def update(self, notice):
        self.company_info = notice.company_info

staff_notice = StaffNotice('初始化公司信息')
staff1 = Staff()
staff2 = Staff()

# staff1和staff2订阅消息
staff_notice.attach(staff1)
staff_notice.attach(staff2)
print(staff1.company_info)
print(staff2.company_info)

print('发布者发布消息1:...')
staff_notice.company_info = '假期放假通知!'
print(staff1.company_info)
print(staff2.company_info)

# staff2不再订阅
staff_notice.detach(staff2)

print('发布者发布消息2:...')
staff_notice.company_info = '明天开会!'
print(staff1.company_info)
print(staff2.company_info)
"""
None
None
发布者发布消息1:...
假期放假通知!
假期放假通知!
发布者发布消息2:...
明天开会!
假期放假通知!
"""

3. 使用场景

当一个抽象模型有两个方面,其中一个方面依赖另一个方面。

将这两者封装在独立对象中以使它们可以各自独立地改变和复用;

当对一个对象的改变需要同时改变其它对象,而不知道具体有多少对象待改变;

当一个对象必须通知其它对象,而它又不能假定其它对象是谁。

换言之,你不希望这些对象是紧耦合的。

优点:

目标和观察者之间的抽象耦合最小;

支持广播通信。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值