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


Posted in Python onJanuary 26, 2016

访问者模式
我觉得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 相关文章推荐
Python ljust rjust center输出
Sep 06 Python
Python标准库之循环器(itertools)介绍
Nov 25 Python
在Python中使用lambda高效操作列表的教程
Apr 24 Python
100行python代码实现跳一跳辅助程序
Jan 15 Python
python+ffmpeg视频并发直播压力测试
Mar 06 Python
解决python2 绘图title,xlabel,ylabel出现中文乱码的问题
Jan 29 Python
在Python中过滤Windows文件名中的非法字符方法
Jun 10 Python
详解Python二维数组与三维数组切片的方法
Jul 18 Python
详解Python并发编程之从性能角度来初探并发编程
Aug 23 Python
Python3操作Excel文件(读写)的简单实例
Sep 02 Python
Django-xadmin后台导入json数据及后台显示信息图标和主题更改方式
Mar 11 Python
用Python提取PDF表格的方法
Apr 11 Python
Python函数中*args和**kwargs来传递变长参数的用法
Jan 26 #Python
python中的编码知识整理汇总
Jan 26 #Python
在MAC上搭建python数据分析开发环境
Jan 26 #Python
python黑魔法之编码转换
Jan 25 #Python
Python编程中对文件和存储器的读写示例
Jan 25 #Python
Python开发如何在ubuntu 15.10 上配置vim
Jan 25 #Python
详解Python验证码识别
Jan 25 #Python
You might like
PHP持久连接mysql_pconnect()函数使用介绍
2012/02/05 PHP
Laravel实现短信注册的示例代码
2018/05/29 PHP
判断滚动条到底部的JS代码
2013/11/04 Javascript
JS 日期比较大小的简单实例
2014/01/13 Javascript
javascript简单实现命名空间效果
2014/03/06 Javascript
Javascript基础教程之函数对象和属性
2015/01/18 Javascript
js仿苹果iwatch外观的计时器代码分享
2015/08/26 Javascript
无需 Flash 使用 jQuery 复制文字到剪贴板
2016/04/26 Javascript
深入解析jQuery中Deferred的deferred.promise()方法
2016/05/03 Javascript
jQuery使用ajax跨域获取数据的简单实例
2016/05/18 Javascript
如何防止INPUT按回车自动提交表单FORM
2016/12/06 Javascript
详解Vue 动态添加模板的几种方法
2017/04/25 Javascript
详解react-router如何实现按需加载
2017/06/15 Javascript
详解vue父子组件间传值(props)
2017/06/29 Javascript
微信小程序实现运动步数排行功能(可删除)
2018/07/05 Javascript
在Vuex使用dispatch和commit来调用mutations的区别详解
2018/09/18 Javascript
使用React手写一个对话框或模态框的方法示例
2019/04/25 Javascript
[27:02]2014 DOTA2国际邀请赛中国区预选赛 5 23 CIS VS LGD第三场
2014/05/24 DOTA
深入浅析Python中的yield关键字
2018/01/24 Python
Pyinstaller将py打包成exe的实例
2018/03/31 Python
Flask框架Flask-Principal基本用法实例分析
2018/07/23 Python
Python文件常见操作实例分析【读写、遍历】
2018/12/10 Python
Python Opencv实现图像轮廓识别功能
2020/03/23 Python
Python八皇后问题解答过程详解
2019/07/29 Python
python 命名规范知识点汇总
2020/02/14 Python
python基于socket函数实现端口扫描
2020/05/28 Python
Python datetime 如何处理时区信息
2020/09/02 Python
前端实现弹幕效果的方法总结(包含css3和canvas的实现方式)
2018/07/12 HTML / CSS
Bailey帽子官方商店:Bailey Hats
2018/09/25 全球购物
当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?
2014/09/09 面试题
好学生评语大全
2014/05/05 职场文书
健康教育评估方案
2014/05/25 职场文书
2014教师年度工作总结
2014/11/10 职场文书
鲁迅故居导游词
2015/02/05 职场文书
小学运动会前导词
2015/07/20 职场文书
JavaScript模拟实现网易云轮播效果
2022/04/04 Javascript