[Python设计模式] 第28章 男人和女人——访问者模式
github地址:https://github.com/cheesezh/python_design_patterns
题目
用程序模拟以下不同情况:
- 男人成功时,背后多半有一个伟大的女人;
- 女人成功时,背后多半有一个失败的男人;
- 男人失败时,闷头喝酒,谁也不用劝;
- 女人失败时,眼泪汪汪,谁也劝不了;
- 男人恋爱时,凡事不懂也要装逼;
- 女人恋爱时,遇事懂也装作不懂;
基础版本
from abc import ABCMeta,abstractmethod
class Person():
__metaclass__ = ABCMeta
def __init__(self):
self.action = None
@abstractmethod
def get_conclusion(self):
pass
class Man(Person):
def get_conclusion(self):
if self.action == "成功":
print("男人成功时,背后多半有一个伟大的女人")
elif self.action == "失败":
print("男人失败时,闷头喝酒,谁也不用劝")
elif self.action == "恋爱":
print("男人恋爱时,凡事不懂也要装逼")
class Woman(Person):
def get_conclusion(self):
if self.action == "成功":
print("女人成功时,背后多半有一个失败的男人")
elif self.action == "失败":
print("女人失败时,眼泪汪汪,谁也劝不了")
elif self.action == "恋爱":
print("女人恋爱时,遇事懂也装作不懂")
def main():
persons = []
man1 = Man()
man1.action = "成功"
persons.append(man1)
woman1 = Woman()
woman1.action = "成功"
persons.append(woman1)
man2 = Man()
man2.action = "失败"
persons.append(man2)
woman2 = Woman()
woman2.action = "失败"
persons.append(woman2)
man3 = Man()
man3.action = "恋爱"
persons.append(man3)
woman3 = Woman()
woman3.action = "恋爱"
persons.append(woman3)
for p in persons:
p.get_conclusion()
main()
男人成功时,背后多半有一个伟大的女人
女人成功时,背后多半有一个失败的男人
男人失败时,闷头喝酒,谁也不用劝
女人失败时,眼泪汪汪,谁也劝不了
男人恋爱时,凡事不懂也要装逼
女人恋爱时,遇事懂也装作不懂
点评
上述代码算是初步的面向对象编程,但是在“Man”和“Woman”中那些“if-else”判断很复杂,如果再增加“结婚”状态,那需要增加分支判断。
这里我们可以使用访问者模式。
访问者模式
访问者模式,表示一个作用于某对象结构中各元素的操作。它使你可以在不改变各元素的类的前提下,定义作用于这些元素的新操作。[DP]
访问者模式的基本代码如下:
from abc import ABCMeta, abstractmethod
class Visitor():
"""
Visitor类,为该对象结构中ConcretElement的每一个类声明一个Visit操作。
"""
__metaclass__ = ABCMeta
@abstractmethod
def visit_concret_element_a(self, concret_element_a):
pass
@abstractmethod
def visit_concret_element_b(self, concret_element_b):
pass
class ConcretVisitor1(Visitor):
"""
ConcretVisitor类,具体访问者,实现每个由Visitor声明的操作。每个操作实现算法的一部分,
而该算法片段乃是对应于结构中对象的类。
"""
def visit_concret_element_a(self, concret_element_a):
print("Visitor1 visit element A")
def visit_concret_element_b(self, concret_element_b):
print("Visitor1 visit element B")
class ConcretVisitor2(Visitor):
"""
同上
"""
def visit_concret_element_a(self, concret_element_a):
print("Visitor2 visit element A")
def visit_concret_element_b(self, concret_element_b):
print("Visitor2 visit element B")
class Element():
"""
Element类,定义一个Accept操作,它以一个访问者为参数
"""
__metaclass__ = ABCMeta
@abstractmethod
def accept(self, visitor):
pass
class ConcretElementA(Element):
"""
具体元素类,实现Accept操作
"""
def accept(self, visitor):
visitor.visit_concret_element_a(self)
def other_functions_a(self):
pass
class ConcretElementB(Element):
"""
具体元素类,实现Accept操作
"""
def accept(self, visitor):
visitor.visit_concret_element_b(self)
def other_functions_b(self):
pass
class ObjectStructure():
"""
ObjectStructure类,能枚举他的元素,可以提供一个高层的接口以允许访问者访问它的元素。
"""
def __init__(self):
self.elements = []
def accept(self, visitor):
for element in self.elements:
element.accept(visitor)
def main():
o = ObjectStructure()
o.elements.append(ConcretElementA())
o.elements.append(ConcretElementB())
v1 = ConcretVisitor1()
v2 = ConcretVisitor2()
o.accept(v1)
o.accept(v2)
main()
Visitor1 visit element A
Visitor1 visit element B
Visitor2 visit element A
Visitor2 visit element B
改进版本——访问者模式
from abc import ABCMeta, abstractmethod
class Action():
"""
状态抽象类,Visitor类,为该对象结构中ConcretElement的每一个类声明一个Visit操作。
"""
__metaclass__ = ABCMeta
@abstractmethod
def get_man_conclusion(self, man):
pass
@abstractmethod
def get_woman_conclusion(self, woman):
pass
class Success(Action):
"""
成功状态类,ConcretVisitor类,具体访问者,实现每个由Visitor声明的操作。每个操作实现算法的一部分,
而该算法片段乃是对应于结构中对象的类。
"""
def get_man_conclusion(self, man):
print("男人成功时,背后多半有一个伟大的女人")
def get_woman_conclusion(self, woman):
print("女人成功时,背后多半有一个失败的男人")
class Failing(Action):
"""
失败状态类,ConcretVisitor类,具体访问者,实现每个由Visitor声明的操作。每个操作实现算法的一部分,
而该算法片段乃是对应于结构中对象的类。
"""
def get_man_conclusion(self, man):
print("男人失败时,闷头喝酒,谁也不用劝")
def get_woman_conclusion(self, woman):
print("女人失败时,眼泪汪汪,谁也劝不了")
class Amativeness(Action):
"""
恋爱状态类,ConcretVisitor类,具体访问者,实现每个由Visitor声明的操作。每个操作实现算法的一部分,
而该算法片段乃是对应于结构中对象的类。
"""
def get_man_conclusion(self, man):
print("男人恋爱时,凡事不懂也要装逼")
def get_woman_conclusion(self, woman):
print("女人恋爱时,遇事懂也装作不懂")
class Person():
"""
人抽象类,Element类,定义一个Accept操作,它以一个访问者为参数
"""
__metaclass__ = ABCMeta
@abstractmethod
def accept(self, visitor):
pass
class Man(Person):
"""
男人类,具体元素类,实现Accept操作
"""
def accept(self, visitor):
"""
双分派技术:
第一次分派:首先在客户端程序中将具体状态作为参数传递给“男人”类完成一次分派;
第二次分派:然后“男人”类调用作为参数的“具体状态”中的方法“男人反应”,同事将自己(self)作为参数传递进去;
"""
visitor.get_man_conclusion(self)
def other_functions_a(self):
pass
class Woman(Person):
"""
女人类,具体元素类,实现Accept操作
"""
def accept(self, visitor):
visitor.get_woman_conclusion(self)
def other_functions_b(self):
pass
class ObjectStructure():
"""
ObjectStructure类,能枚举他的元素,可以提供一个高层的接口以允许访问者访问它的元素。
"""
def __init__(self):
self.elements = []
def accept(self, visitor):
for element in self.elements:
element.accept(visitor)
def main():
o = ObjectStructure()
o.elements.append(Man())
o.elements.append(Woman())
v1 = Success()
v2 = Failing()
v3 = Amativeness()
o.accept(v1)
print("-"*10)
o.accept(v2)
print("-"*10)
o.accept(v3)
main()
print()
print("由于使用了双分派技术,当新增`结婚`状态时,只需要新增一个状态子类,然后修改客户端即可。")
print()
class Marriage(Action):
def get_man_conclusion(self, man):
print("男人结婚时,有妻徒刑遥无期")
def get_woman_conclusion(self, woman):
print("女人结婚时,婚姻保险保平安")
def main():
o = ObjectStructure()
o.elements.append(Man())
o.elements.append(Woman())
v1 = Success()
v2 = Failing()
v3 = Amativeness()
v4 = Marriage()
o.accept(v1)
print("-"*10)
o.accept(v2)
print("-"*10)
o.accept(v3)
print("-"*10)
o.accept(v4)
main()
男人成功时,背后多半有一个伟大的女人
女人成功时,背后多半有一个失败的男人
----------
男人失败时,闷头喝酒,谁也不用劝
女人失败时,眼泪汪汪,谁也劝不了
----------
男人恋爱时,凡事不懂也要装逼
女人恋爱时,遇事懂也装作不懂
由于使用了双分派技术,当新增`结婚`状态时,只需要新增一个状态子类,然后修改客户端即可。
男人成功时,背后多半有一个伟大的女人
女人成功时,背后多半有一个失败的男人
----------
男人失败时,闷头喝酒,谁也不用劝
女人失败时,眼泪汪汪,谁也劝不了
----------
男人恋爱时,凡事不懂也要装逼
女人恋爱时,遇事懂也装作不懂
----------
男人结婚时,有妻徒刑遥无期
女人结婚时,婚姻保险保平安
点评
访问者模式适用于数据结构相对稳定的系统,它把数据结构和作用于数据结构上的操作之间的耦合解脱开,使得操作结合可以相对自由地演化。
访问这模式的目的是要把处理从数据结构分离出来。很多系统可以按照算法和数据结构分开,如果这样的系统有比较稳定的数据结构,又有易于变化的算法的话,那就比较适合使用访问者模式,因为访问者模式使得算法操作的增加变得容易。反之,如果系统的数据结构对象易于变化,经常要有新的数据对象增加进来,就不适合使用访问者模式。
访问者模式的优点就是增加新的操作很容易,因为增加新的操作就意味着增加一个新的访问者。访问者模式将有关的行为集中到一个访问者对象中。
访问者模式的缺点其实就是增加新的数据结构变得困难了。
GoF四人中的一个作者说过,大多数时候并不需要访问者模式,但当一旦需要访问者模式时,那就是真的需要它了。因为我们很难找到数据结构不变化的情况,所以用访问者模式的机会也就不太多。
[Python设计模式] 第28章 男人和女人——访问者模式的更多相关文章
- [Python设计模式] 第24章 加薪审批——职责链模式
github地址:https://github.com/cheesezh/python_design_patterns 题目 用程序模拟以下情景 员工向经理发起加薪申请,经理无权决定,需要向总监汇报, ...
- [Python设计模式] 第20章 挨个买票——迭代器模式
github地址:https://github.com/cheesezh/python_design_patterns 迭代器模式 迭代器模式,提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该 ...
- [Python设计模式] 第18章 游戏角色备份——备忘录模式
github地址:https://github.com/cheesezh/python_design_patterns 题目 用代码模拟以下场景,一个游戏角色有生命力,攻击力,防御力等数据,在打Bos ...
- [Python设计模式] 第8章 学习雷锋好榜样——工厂方法模式
github地址:https://github.com/cheesezh/python_design_patterns 简单工厂模式 v.s. 工厂方法模式 以简单计算器为例,对比一下简单工厂模式和工 ...
- [Python设计模式] 第21章 计划生育——单例模式
github地址:https://github.com/cheesezh/python_design_patterns 单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式 ...
- [Python设计模式] 第1章 计算器——简单工厂模式
github地址:https://github.com/cheesezh/python_design_patterns 写在前面的话 """ 读书的时候上过<设计模 ...
- [Python设计模式] 第22章 手机型号&软件版本——桥接模式
github地址:https://github.com/cheesezh/python_design_patterns 紧耦合程序演化 题目1 编程模拟以下情景,有一个N品牌手机,在上边玩一个小游戏. ...
- [Python设计模式] 第9章 如何准备多份简历——原型模式
github地址:https://github.com/cheesezh/python_design_patterns 题目 设计一个简历类,必须有姓名,可以设置性别和年龄,即个人信息,可以设置曾就职 ...
- [Python设计模式] 第7章 找人帮忙追美眉——代理模式
github地址:https://github.com/cheesezh/python_design_patterns 题目1 Boy追求Girl,给Girl送鲜花,送巧克力,送洋娃娃. class ...
随机推荐
- Django1.11.7配置静态文件
Django配置静态文件分为三步 1.建文件夹 2.设置setting 3.页面引用 1.文件目录结构 在APP下新建static文件夹,将js和css文件放入文件夹 2.配置settings.py ...
- input按钮去掉默认样式
border: 1px solid transparent; //自定义边框 outline: none; //消除默认点击蓝色边框效果
- 2018年中国研究生数学建模竞赛C题 二等奖 赛题论文
2018年中国研究生数学建模竞赛C题 对恐怖袭击事件记录数据的量化分析 恐怖袭击是指极端分子或组织人为制造的.针对但不仅限于平民及民用设施的.不符合国际道义的攻击行为,它不仅具有极大的杀伤性与破坏力, ...
- 四、Python导入自己写的包报错:没有该包如何解决
场景:当你运行文件时,提示没有你想要导入的这个包,这是只需在你运行文件的开头添加俩行代码: import sys sys.path.apprnd('../') 原理:请看图解 另外解释下sys.pat ...
- POJ 2823 Sliding Window (模板题)【单调队列】
<题目链接> <转载于>>> > 题目大意: 给你一段序列和一个长为k的窗口,这个窗口从最左边逐渐向右滑,直到滑到最右边,问你,该窗口在滑动的过程中,最大值和 ...
- datatables数据渲染自定义
"data": "ip",渲染回调函数中的data['ip']字段将传给render函数中的data:render函数的返回内容将代替"data&qu ...
- 【可靠性】Mysql 5.7 降低了半同步复制-数据丢失的风险
如果你的生产线开启了半同步复制,那么对数据的一致性会要求较高,但在MySQL5.5/5.6里,会存在数据不一致的风险.有这么一个场景,客户端提交了一个事务,master把binlog发送给slave, ...
- XamarinSQLite教程Xamarin.iOS项目中打开数据库文件
XamarinSQLite教程Xamarin.iOS项目中打开数据库文件 以下是打开MyDocuments.db数据库的具体操作步骤: (1)将Mac电脑上的MyDocuments.db数据库移动到W ...
- Navicat -- 远程连接问题
有朋友可能会碰到使用Navicat for mysql 远程连接 mySql数据库会提示10061.1045错误或 2003-Can’t connect to MySQL on ’192.168.1. ...
- 一道有意思的找规律题目 --- CodeForces - 964A
题目连接: https://vjudge.net/problem/1502082/origin 这一题第一眼看过去貌似是模拟,但是根据其范围是1e9可以知道,如果暴力解基本上是不可能的(不排除大佬级优 ...