19、Python - 访问者模式
基本介绍
访问者模式(Visitor Pattern)可以让多个处理对象对同一个数据对象做出不同的反应,但并不会改变被访问的数据对象。
特点:主要将数据结构与数据操作分离
访问者模式应该是所有模式中最抽象、最难以理解的一个。
该模式属于行为型模式。
案例图示
上市公司的原始财务数据,对于会计来说需要制作各种报表,对于财务总监来说需要分析公司业绩,对于战略顾问来说需要分析行业变化。
它们总是对同一份数据进行访问,但是获得的结果都是不同的:
优缺点
优点:
- 符合单一职责原则
- 优秀的扩展性
- 灵活性
缺点:
- 具体元素对访问者公布细节,违反了迪米特原则
- 具体元素变更比较困难
- 违反了依赖倒置原则,依赖了具体类,没有依赖抽象
代码实现
用Python实现访问者模式:
class Finance:
"""财务数据结构类"""
def __init__(self):
销售额
self.salesvolume = None
成本
self.cost = None
历史销售额
self.history_salesvolume = None
历史成本
self.history_cost = None
def set_salesvolume(self, value):
self.salesvolume = value
def set_cost(self, value):
self.cost = value
def set_history_salesvolume(self, value):
self.history_salesvolume = value
def set_history_cost(self, value):
self.history_cost = value
def accept(self, visitor):
pass
class Finance_year(Finance):
"""2018年财务数据类"""
def __init__(self, year):
Finance.__init__(self)
self.work = [] 安排工作人员列表
self.year = year
def add_work(self, work):
self.work.append(work)
def accept(self):
for obj in self.work:
obj.visit(self)
class Accounting:
"""会计类"""
def __init__(self):
self.ID = "会计"
self.Duty = "计算报表"
def visit(self, table):
print('会计年度: {}'.format(table.year))
print("我的身份是: {} 职责: {}".format(self.ID, self.Duty))
print('本年度纯利润: {}'.format(table.salesvolume - table.cost))
print('------------------')
class Audit:
"""财务总监类"""
def __init__(self):
self.ID = "财务总监"
self.Duty = "分析业绩"
def visit(self, table):
print('会计总监年度: {}'.format(table.year))
print("我的身份是: {} 职责: {}".format(self.ID, self.Duty))
if table.salesvolume - table.cost > table.history_salesvolume - table.history_cost:
msg = "较同期上涨"
else:
msg = "较同期下跌"
print('本年度公司业绩: {}'.format(msg))
print('------------------')
class Adviser:
"""战略顾问"""
def __init__(self):
self.ID = "战略顾问"
self.Duty = "制定明年战略"
def visit(self, table):
print('战略顾问年度: {}'.format(table.year))
print("我的身份是: {} 职责: {}".format(self.ID, self.Duty))
if table.salesvolume > table.history_salesvolume:
msg = "行业上行,扩大生产规模"
else:
msg = "行业下行,减小生产规模"
print('本年度公司业绩: {}'.format(msg))
print('------------------')
class Work:
"""工作类"""
def __init__(self):
self.works = [] 需要处理的年度数据列表
def add_work(self, obj):
self.works.append(obj)
def remove_work(self, obj):
self.works.remove(obj)
def visit(self):
for obj in self.works:
obj.accept()
if __name__ == '__main__':
work = Work() 计划安排财务、总监、顾问对2018年数据处理
实例化2018年数据结构
finance_2018 = Finance_year(2018)
finance_2018.set_salesvolume(200)
finance_2018.set_cost(100)
finance_2018.set_history_salesvolume(180)
finance_2018.set_history_cost(90)
实例化会计
accounting = Accounting()
实例化总监
audit = Audit()
实例化总监
adviser = Adviser()
会计安排到2018分析日程中
finance_2018.add_work(accounting)
finance_2018.add_work(audit)
顾问安排到2018分析日程中
finance_2018.add_work(adviser)
添加2018年财务工作安排
work.add_work(finance_2018)
work.visit()
# 会计年度: 2018
# 我的身份是: 会计 职责: 计算报表
# 本年度纯利润: 100
# ------------------
# 会计总监年度: 2018
# 我的身份是: 财务总监 职责: 分析业绩
# 本年度公司业绩: 较同期上涨
# ------------------
# 战略顾问年度: 2018
# 我的身份是: 战略顾问 职责: 制定明年战略
# 本年度公司业绩: 行业上行,扩大生产规模
# ------------------