Python设计模式之观察者模式

1.定义

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

2.动机

将一个系统分割成一系列相互协作的类有一个常见的副作用:需要维护相关对象间的一致性。我们不希望为了维持一致性而使各类紧密耦合,因为这样降低了它们的可重用性。

3.适用性

  • 当一个抽象模型有两个方面 , 其中一个方面依赖于另一方面。将这二者封装在独立的对象中以使它们可以各自独立地改变和复用。
  • 当对一个对象的改变需要同时改变其它对象 , 而不知道具体有多少对象有待改变。
  • 当一个对象必须通知其它对象,而它又不能假定其它对象是谁。换言之 , 你不希望这些对象是紧密耦合的。

4.优缺点

  • 目标和观察者间的抽象耦合
  • 支持广播通信
  • 意外的更新

5.具体实现

# -*- coding: utf-8 -*-
'''
------------------------------------------------------------
File Name: Observer.py
Description : 
Project: Test
Last Modified: Monday, 18th March 2019 7:59:03 pm
-------------------------------------------------------------
'''
from typing import (
    List,
    Any,
    NoReturn
)
from dataclasses import (
    dataclass,
    field
)


@dataclass
class Subject:
    _observers: List[Any] = field(default_factory=list)

    def attach(self, observer: object) -> NoReturn:
        if observer not in self._observers:
            self._observers.append(observer)

    def detach(self, observer: object) -> NoReturn:
        try:
            self._observers.remove(observer)
        except ValueError:
            pass

    def notify(self, modifier=None) -> NoReturn:
        for observer in self._observers:
            if modifier != observer:
                observer.update(self)


# Example usage
@dataclass
class Data(Subject):
    name: str = ""
    _data: int = 0

    @property
    def data(self) -> Any:
        return self._data

    @data.setter
    def data(self, value) -> NoReturn:
        self._data = value
        self.notify()


class HexViewer:

    def update(self, subject) -> NoReturn:
        print(u'HexViewer: Subject %s has data 0x%x' %
              (subject.name, subject.data))


class DecimalViewer:

    def update(self, subject) -> NoReturn:
        print(u'DecimalViewer: Subject %s has data %d' %
              (subject.name, subject.data))


# Example usage...
def main() -> NoReturn:
    data1 = Data(name='Data 1')
    data2 = Data(name='Data 2')
    view1 = DecimalViewer()
    view2 = HexViewer()
    data1.attach(view1)
    data1.attach(view2)
    data2.attach(view2)
    data2.attach(view1)

    print(u"Setting Data 1 = 10")
    data1.data = 10
    print(u"Setting Data 2 = 15")
    data2.data = 15
    print(u"Setting Data 1 = 3")
    data1.data = 3
    print(u"Setting Data 2 = 5")
    data2.data = 5
    print(u"Detach HexViewer from data1 and data2.")
    data1.detach(view2)
    data2.detach(view2)
    print(u"Setting Data 1 = 10")
    data1.data = 10
    print(u"Setting Data 2 = 15")
    data2.data = 15


if __name__ == '__main__':
    main()

# Output

Setting Data 1 = 10
DecimalViewer: Subject Data 1 has data 10
HexViewer: Subject Data 1 has data 0xa
Setting Data 2 = 15
HexViewer: Subject Data 2 has data 0xf
DecimalViewer: Subject Data 2 has data 15
Setting Data 1 = 3
DecimalViewer: Subject Data 1 has data 3
HexViewer: Subject Data 1 has data 0x3
Setting Data 2 = 5
HexViewer: Subject Data 2 has data 0x5
DecimalViewer: Subject Data 2 has data 5
Detach HexViewer from data1 and data2.
Setting Data 1 = 10
DecimalViewer: Subject Data 1 has data 10
Setting Data 2 = 15
DecimalViewer: Subject Data 2 has data 15
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 观察者模式 有时,我们希望在一个对象的状态改变时更新另外一组对象。在MVC模式中有这样一个非常常见的例子,假设在两...
    英武阅读 4,232评论 0 51
  • 推荐深度好文:原文链接,作者:张洋 序言 Perface “佛曰:苦海无涯,回头是岸。——佛教用语” 面向对象(O...
    不知公民阅读 790评论 0 4
  • 序言 “佛曰:苦海无涯,回头是岸。——佛教用语” 面向对象(Object-Oriented),这是一条令无数开发人...
    叶小健阅读 2,435评论 4 24
  • 设计模式基本原则 开放-封闭原则(OCP),是说软件实体(类、模块、函数等等)应该可以拓展,但是不可修改。开-闭原...
    西山薄凉阅读 3,856评论 3 14
  • 即兴演讲有四个要点:立马举例、联系自己、说出感受、号召呼吁,最重要的是要演说时要与他人连接,如何做到连接他人,关键...
    黄VIVI阅读 151评论 0 0