举例讲解Python设计模式编程中的访问者与观察者模式

作者:dongweiming 时间:2021-06-10 06:38:16 

访问者模式
我觉得Visitor模式是在补修改已有程序结构前提下,通过添加额外的访问者完成对代码功能的拓展 为什么这样用?当你的类层次较多,在某层结构中增加新的方法,要是在基类上面添加或者变更,可能破坏原来的设计, 有兼容问题,所以只在需要的类上面动态添加。

python的例子
这里是个构建车的例子,每个部件都有一个accept的方法接受我上面说的所谓'访问者',而这个访问者 以参数的方式传进来,但是其实他是一个含有一些功能的类的实例,它拥有很多个visit开头的方法对应不同的部件。 这样就不需要修改这些部件,而只是修改我们的访问者类的相关部分。


# 轮子,引擎, 车身这些定义好了都不需要变动
class Wheel:
 def __init__(self, name):
   self.name = name
 def accept(self, visitor):
   # 每个visitor是同样的,但是其中的方法是不一样的,比如这里是visitWheel,
   # 然后传入了self,想想?他其实想做什么就能做什么
   visitor.visitWheel(self)

class Engine:
 def accept(self, visitor):
   visitor.visitEngine(self)

class Body:
 def accept(self, visitor):
   visitor.visitBody(self)

# 我们要组合成车
class Car:
 def __init__(self):
   self.engine = Engine()
   self.body  = Body()
   self.wheels = [ Wheel("front left"), Wheel("front right"),
           Wheel("back left") , Wheel("back right") ]

# 这个也不需要在动,他只是上面部件的组合,只是做了属性的委托
 def accept(self,visitor):
   visitor.visitCar(self)
   self.engine.accept(visitor)
   self.body.accept(visitor)
   for wheel in self.wheels:
     wheel.accept(visitor)

# 这个才是我们的访问者,每次的修改都在这里面
class PrintVisitor:
 def visitWheel(self, wheel):
   print "Visiting "+wheel.name+" wheel"
 def visitEngine(self, engine):
   print "Visiting engine"
 def visitBody(self, body):
   print "Visiting body"
 def visitCar(self, car):
   print "Visiting car"

if __name__ == '__main__':
 car = Car()
 visitor = PrintVisitor()
 car.accept(visitor)


观察者模式
当我们希望一个对象的状态发生变化,那么依赖与它的所有对象都能相应变化(获得通知),那么就可以用到Observer模式, 其中的这些依赖对象就是观察者的对象,那个要发生变化的对象就是所谓'观察者'

python的例子


# 这个是观察者基类
class Subject(object):
 def __init__(self):
   self._observers = []

# 添加依赖的对象
 def attach(self, observer):
   if not observer in self._observers:
     self._observers.append(observer)

# 取消添加
 def detach(self, observer):
   try:
     self._observers.remove(observer)
   except ValueError:
     pass

# 这里只是通知上面注册的依赖对象新的变化
 def notify(self, modifier=None):
   for observer in self._observers:
     # 可以设置过滤条件,对不符合过滤条件的更新
     if modifier != observer:
       observer.update(self)

# 观察者类
class Data(Subject):
 def __init__(self, name=''):
   super(Data, self).__init__()
   self.name = name
   self._data = 0

# python2.6新增的写法,获取属性为property,设置属性为(假设属性名字为x)@x.setter,删除为@x.deleter
 @property
 def data(self):
   return self._data

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

# 这里有2个被观察者,也就是依赖的对象,每次Data有改变,这2个view都会变动
class HexViewer(object):
 def update(self, subject):
   print 'HexViewer: Subject %s has data 0x%x' % (subject.name, subject.data)

class DecimalViewer(object):
 def update(self, subject):
   print 'DecimalViewer: Subject %s has data %d' % (subject.name, subject.data)

if __name__ == '__main__':

data1 = Data('Data 1')
 data2 = Data('Data 2')
 view1 = DecimalViewer()
 view2 = HexViewer()
 data1.attach(view1)
 data1.attach(view2)
 data2.attach(view2)
 data2.attach(view1)

print "Setting Data 1 = 10"
 data1.data = 10
 print "Setting Data 2 = 15"
 data2.data = 15
 print "Setting Data 1 = 3"
 data1.data = 3
 print "Setting Data 2 = 5"
 data2.data = 5
 print "Update data1's view2 Because view1 is be filtered"
 data1.notify(modifier=view1)
 print "Detach HexViewer from data1 and data2."
 data1.detach(view2)
 data2.detach(view2)
 print "Setting Data 1 = 10"
 data1.data = 10
 print "Setting Data 2 = 15"
 data2.data = 15

标签:Python,设计模式
0
投稿

猜你喜欢

  • 分享4个Python中的非常好用的自动化脚本

    2023-12-03 09:59:17
  • MySQL 函数过程递归

    2008-07-25 19:32:00
  • 你是真正的用户体验设计者吗? Ⅴ

    2008-04-19 18:32:00
  • python中set()函数简介及实例解析

    2022-05-15 17:12:24
  • pandas带有重复索引操作方法

    2023-04-19 20:22:12
  • ASP+SQLServer2000 经验积累

    2008-02-03 15:16:00
  • 如何利用python发送邮件

    2022-11-09 09:34:31
  • python实现K折交叉验证

    2023-06-08 18:49:01
  • python密码学周期置换密码学习

    2021-09-21 16:45:39
  • 使用Dreamweaver便捷技巧方法十六则

    2010-07-02 16:28:00
  • 解决linux下使用python打开terminal时报错的问题

    2022-10-06 10:46:25
  • Python密码学仿射密码及攻击单字母密码教程

    2021-01-25 05:52:28
  • Python自然语言处理之词干,词形与最大匹配算法代码详解

    2023-07-23 04:48:37
  • Python中super函数用法实例分析

    2023-12-05 14:36:19
  • python之语句mode = 'test' if y is None else 'train'问题

    2021-02-04 20:35:17
  • Python实现的栈、队列、文件目录遍历操作示例

    2022-06-10 00:12:35
  • Python对切片命名的实现方法

    2023-09-30 09:43:15
  • 详解Python中映射类型(字典)操作符的概念和使用

    2021-12-05 09:24:43
  • 在python中利用try..except来代替if..else的用法

    2023-09-12 17:50:10
  • PHP hex2bin()函数用法讲解

    2023-06-06 18:51:35
  • asp之家 网络编程 m.aspxhome.com