举例讲解Python设计模式编程中的访问者与观察者模式
作者:dongweiming 发布时间:2021-06-10 06:38:16
标签:Python,设计模式
访问者模式
我觉得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
0
投稿
猜你喜欢
- 本文实例讲述了Python实现的凯撒密码算法。分享给大家供大家参考,具体如下:一 介绍凯撒密码是一种非常古老的加密方法,相传当年凯撒大地行军
- 微软 Office 提供基于 COM 接口的编程。Python 通过 pywin32 可以方便地调用各组件。如果下载和安装 pywin32
- 三元条件判断的3种实现方法C语言中有三元条件表达式,如 a>b?a:b,Python中没有三目运算符(?:),但Python有它自己的
- 本文介绍了python画图时设置分辨率和画布大小的实现,主要使用plt.figure(),下面就一起来了解一下plt.figure()示例:
- 进程是什么?进程就是一个程序在一个数据集上的一次动态执行过程。进程一般由程序、数据集、进程控制块三部分组成。我们编写的程序用来描述进程要完成
- 这篇文章主要介绍了python 利用jinja2模板生成html代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考
- pytorch设置随机种子pytorch设置随机种子 - 保证复现模型所有的训练过程在使用 PyTorch 时,如果希望通过设置随机数种子,
- 1.彻底弄懂CSS盒子模式一(DIV布局快速入门) 2.彻底弄懂CSS盒子模式二(导航栏实例) 4.彻底弄懂CSS盒子模式四(绝对定位和相对
- 开篇:测试过程中,对于多参数参数多值的情况进行测试用例组织,之前一直使用【正交分析法】进行用例组织,说白了就是把每个参数的所有值分别和其他参
- 变量不需要声明Python的变量不需要声明,你可以直接输入:>>>a = 10那么你的内存里就有了一个变量a, 它的值是1
- 今天写了一个获取当前公网ip并且自动断开宽带连接的文件,和大家分享下。这个文件的具体用途大家懂的,可以尽管拿去用,不过目前只适用于Windo
- 描述Python time time()返回当前时间的时间戳(1970 * 后经过的浮点秒数)。突然想看时间了,打开cmd发现脑中空荡,Jav
- asp之家注:本文介绍的长文章分页方法不错,作者分析的很详细,用分页符来手动为长文章分页,应该是最好的长文章分页方法,我们不必担心会把一些代
- 代码实例:try: import termios, TERMIOS 1except ImportErro
- Access 操作很简单,具体不步骤如下:打开你mdb数据库,工具-->数据库实用工具-->压缩和修复数据库(c)... SQL SERVE
- 刚才运行了一段代码,来查看Request.ServerVariables里面有多少值,看了一下,共50个!代码<%=Request.S
- 正常来说backward( )函数是要传入参数的,一直没弄明白backward需要传入的参数具体含义,但是没关系,生命在与折腾,咱们来折腾一
- 看代码吧~# 加载库import numpy as npfrom fancyimpute import KNNfrom sklearn.pr
- 一、为何使用Tkinter而非PyQt众所周知,在Python中创建图形界面程序有很多种的选择,其中PyQt和wxPython都是很热门的模
- python实现12306余票查询我们说先在浏览器中打开开发者工具(F12),尝试一次余票的查询,通过开发者工具查看发出请求的包余票查询界面