observer.py 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. """http://code.activestate.com/recipes/131499-observer-pattern/"""
  2. class Subject(object):
  3. def __init__(self):
  4. self._observers = []
  5. def attach(self, observer):
  6. if not observer in self._observers:
  7. self._observers.append(observer)
  8. def detach(self, observer):
  9. try:
  10. self._observers.remove(observer)
  11. except ValueError:
  12. pass
  13. def notify(self, modifier=None):
  14. for observer in self._observers:
  15. if modifier != observer:
  16. observer.update(self)
  17. # Example usage
  18. class Data(Subject):
  19. def __init__(self, name=''):
  20. Subject.__init__(self)
  21. self.name = name
  22. self._data = 0
  23. @property
  24. def data(self):
  25. return self._data
  26. @data.setter
  27. def data(self, value):
  28. self._data = value
  29. self.notify()
  30. class HexViewer:
  31. def update(self, subject):
  32. print('HexViewer: Subject %s has data 0x%x' %
  33. (subject.name, subject.data))
  34. class DecimalViewer:
  35. def update(self, subject):
  36. print('DecimalViewer: Subject %s has data %d' %
  37. (subject.name, subject.data))
  38. # Example usage...
  39. def main():
  40. data1 = Data('Data 1')
  41. data2 = Data('Data 2')
  42. view1 = DecimalViewer()
  43. view2 = HexViewer()
  44. data1.attach(view1)
  45. data1.attach(view2)
  46. data2.attach(view2)
  47. data2.attach(view1)
  48. print("Setting Data 1 = 10")
  49. data1.data = 10
  50. print("Setting Data 2 = 15")
  51. data2.data = 15
  52. print("Setting Data 1 = 3")
  53. data1.data = 3
  54. print("Setting Data 2 = 5")
  55. data2.data = 5
  56. print("Detach HexViewer from data1 and data2.")
  57. data1.detach(view2)
  58. data2.detach(view2)
  59. print("Setting Data 1 = 10")
  60. data1.data = 10
  61. print("Setting Data 2 = 15")
  62. data2.data = 15
  63. if __name__ == '__main__':
  64. main()