美文网首页
Python设计模式之观察者模式

Python设计模式之观察者模式

作者: 云爬虫技术研究笔记 | 来源:发表于2019-03-18 22:59 被阅读0次

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

相关文章

网友评论

      本文标题:Python设计模式之观察者模式

      本文链接:https://www.haomeiwen.com/subject/hmwimqtx.html