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
网友评论