Design 访问者模式

基本介绍

在访问者模式(Visitor Pattern)中,我们使用了一个访问者类,它改变了元素类的执行算法。

通过这种方式,元素的执行算法可以随着访问者改变而改变。这种类型的设计模式属于行为型模式。

根据模式,元素对象已接受访问者对象,这样访问者对象就可以处理元素对象上的操作。

特点:主要将数据结构与数据操作分离

访问者模式应该是所有模式中最抽象、最难以理解的一个。

案例图示

上市公司的原始财务数据,对于会计来说需要制作各种报表,对于财务总监来说需要分析公司业绩,对于战略顾问来说需要分析行业变化:

image-20210414145127124

优缺点

优点:

  • 符合单一职责原则
  • 优秀的扩展性
  • 灵活性

缺点:

  • 具体元素对访问者公布细节,违反了迪米特原则
  • 具体元素变更比较困难
  • 违反了依赖倒置原则,依赖了具体类,没有依赖抽象

Python实现

用Python实现--模式:

#! /usr/local/bin/python3
# -*- coding:utf-8 -*-

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
我的身份是: 战略顾问 职责: 制定明年战略
本年度公司业绩: 行业上行,扩大生产规模
------------------

Golang实现

用Golang实现访问者模式:

...
posted @ 2021-04-14 14:53  云崖君  阅读(42)  评论(0编辑  收藏  举报