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章 男人和女人——访问者模式的更多相关文章

  1. [Python设计模式] 第24章 加薪审批——职责链模式

    github地址:https://github.com/cheesezh/python_design_patterns 题目 用程序模拟以下情景 员工向经理发起加薪申请,经理无权决定,需要向总监汇报, ...

  2. [Python设计模式] 第20章 挨个买票——迭代器模式

    github地址:https://github.com/cheesezh/python_design_patterns 迭代器模式 迭代器模式,提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该 ...

  3. [Python设计模式] 第18章 游戏角色备份——备忘录模式

    github地址:https://github.com/cheesezh/python_design_patterns 题目 用代码模拟以下场景,一个游戏角色有生命力,攻击力,防御力等数据,在打Bos ...

  4. [Python设计模式] 第8章 学习雷锋好榜样——工厂方法模式

    github地址:https://github.com/cheesezh/python_design_patterns 简单工厂模式 v.s. 工厂方法模式 以简单计算器为例,对比一下简单工厂模式和工 ...

  5. [Python设计模式] 第21章 计划生育——单例模式

    github地址:https://github.com/cheesezh/python_design_patterns 单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式 ...

  6. [Python设计模式] 第1章 计算器——简单工厂模式

    github地址:https://github.com/cheesezh/python_design_patterns 写在前面的话 """ 读书的时候上过<设计模 ...

  7. [Python设计模式] 第22章 手机型号&软件版本——桥接模式

    github地址:https://github.com/cheesezh/python_design_patterns 紧耦合程序演化 题目1 编程模拟以下情景,有一个N品牌手机,在上边玩一个小游戏. ...

  8. [Python设计模式] 第9章 如何准备多份简历——原型模式

    github地址:https://github.com/cheesezh/python_design_patterns 题目 设计一个简历类,必须有姓名,可以设置性别和年龄,即个人信息,可以设置曾就职 ...

  9. [Python设计模式] 第7章 找人帮忙追美眉——代理模式

    github地址:https://github.com/cheesezh/python_design_patterns 题目1 Boy追求Girl,给Girl送鲜花,送巧克力,送洋娃娃. class ...

随机推荐

  1. BZOJ3172 [Tjoi2013]单词 字符串 SA ST表

    原文链接http://www.cnblogs.com/zhouzhendong/p/9026543.html 题目传送门 - BZOJ3172 题意 输入$n(n\leq 200)$个字符串,保证长度 ...

  2. 第一章:python基础语法| 字符编码| 条件语句...

    1.编程语言介绍 编程就是写代码,让计算机帮你做事情.计算机底层是电路,只认识二进制0和1.机器语言&汇编语言语言进化历史:机器.汇编.高级.机器语言只接受二进制代码:汇编语言是采用英文缩写的 ...

  3. Lunch War with the Donkey CSU - 2084

    Jingze is a big figure in California State University for his stubbornness. Because of his new failu ...

  4. HashMap 源码阅读

    前言 之前读过一些类的源码,近来发现都忘了,再读一遍整理记录一下.这次读的是 JDK 11 的代码,贴上来的源码会去掉大部分的注释, 也会加上一些自己的理解. Map 接口 这里提一下 Map 接口与 ...

  5. Ignatius and the Princess III HDU - 1028 -生成函数or完全背包计数

    HDU - 1028 step 1:初始化第一个多项式 也就是 由 1的各种方案 组 成 的多项式 初始化系数为 1.临时区 temp初始化 为 0 step 2:遍历后续的n - 1 个 多项式 , ...

  6. spring框架加载完成后执行上下文刷新事件(ContextRefreshedEvent)

    目前spring框架是j2ee比较常用的项目开发技术,只需在web.xml文件中进行少许配置即可,代码如下所示:<!--spring的配置文件--><context-param> ...

  7. vue中的页面渲染方案

    一.模板渲染 <div id="J_render_app"> <ul v-if="items.length"> <li v-for ...

  8. pandas 基本操作

    1.     一维数据结构Series a.   概念:Series 是pandas 的一维数据结构,有重要的两个属性 index 和values b.  初始化: 可以通过 python 的 Lis ...

  9. 网络基础配置--开启SSH,关闭Telnet

    1.Telnet和SSH对比 1.1.TELNET 使用Telnet这个用来访问远程计算机的TCP/IP协议以控制你的网络设备相当于在离开某个建筑时大喊你的用户名和口令.很快会有人进行监听,并且他们会 ...

  10. Python基础知识—sys模块初探

    有关Python解释器的信息 与所有其他模块一样,必须使用import语句导入sys模块,即import sys. sys模块提供有关Python解释器的常量,函数和方法.dir(系统)给出了可用常量 ...