什么是访问者模式?

一个对象有稳定的数据结构,却为不同的访问者提供不同的数据操作,对象提供接收访问者的方法,从而保证数据结构的稳定性和操作的多样性。也可以理解为,封装对象的操作方法,达到不改变对象数据结构的稳定性同时易于扩展操作。

解决的主要问题

主要解决:稳定的数据结构和易变的操作耦合问题。

如何实现

(1)Visitor接口:访问者接口,封装对象元素的操作,它定义了对每一个元素(Element)访问的行为,它的参数就是可以访问的元素,它的方法数理论上来讲与元素个数是一样的。

(2)Visitor1、Visitor2:访问者 -- 具体的访问类,它需要给出对每一个元素类访问时所产生的具体行为,如老板,会计。

(3)Element:元素对外访问入口接口,它定义了一个接受访问者的方法(Accept),其意义是指每一个元素(子类)都要可以被访问者访问。

(4)ElementA、ElementB:具体的元素类,它提供接受访问方法的具体实现,而这个具体的实现,通常情况下是使用访问者提供的访问该元素类的方法。

(5)Object:定义当中所说的对象结构,对象结构是一个抽象表述,它内部管理了元素集合,并且可以迭代这些元素供访问者访问
生活场景

公司有一个账本,抽象为一个对象,它有两个稳定的元素,一个是收入,一个是支出。公司有不同的角色都需要访问账本,抽象为访问者。比如。老板只关注总收入和总支出,会计关心每一笔收入是否缴税,每发一笔工资是否扣税。两者间是不同操作,且还有以后可能其他角色需要访问账本。

demo代码

1、Visitor接口

/**
* 访问者接口
* (1)查看收入账单
* (1)查看之处账单
*/
public interface IAccountBookViewer {
/**
* 消费账单
* @param bill
*/
void view(ConsumeBill bill); /**
* 收入账单
* @param bill
*/
void view(IncomeBill bill); }

IAccountBookViewer.java

2、访问者

2.1 具体类 --  老板

 /**
* 老板角色--查看账单
* 老板只关注总收入和总支出
*/
public class Boss implements IAccountBookViewer { /**
* 总消费
*/
private Double totalConsume = 0d;//默认值是null /**
* 总收入
*/
private Double totalIncome = 0d;//默认值是null public Double getTotalConsume() {
System.out.println("老板查看总支出:" + totalConsume);
return totalConsume;
} public Double getTotalIncome() {
System.out.println("老板查看总收入:" + totalIncome);
return totalIncome;
} /**
* 查看消费账单
*
* @param bill 消费账单
*/
@Override
public void view(ConsumeBill bill) {
totalConsume += bill.getAmount();
} /**
* 查看收入账单
*
* @param bill 账单
*/
@Override
public void view(IncomeBill bill) {
totalIncome += bill.getAmount();
}
}

Boss.java

2.2、具体类 -- 会计

 /**
* 注册会计师
* 关注具体收入和支出是否交税
*/
public class CPA implements IAccountBookViewer {
/**
* 查看支出,如果是工资,是否已经交税
*
* @param bill
*/
@Override
public void view(ConsumeBill bill) {
if (bill.getItem().equals("工资")) {
System.out.println("CPA查看是否工资已经扣税");
}
} /**
* 查看收入,所有收入都要交税
*
* @param bill
*/
@Override
public void view(IncomeBill bill) {
System.out.println("CPA查看所有收入是否已经缴税");
}
}

CPA.java

3、元素接口,定义每个元素行为 -- 对外提供accept方法,传入访问者

 /**
* 账单类接口,接收访问者
*/
public interface IBill {
void accept(IAccountBookViewer viewer);
}

IBill.java

4、具体的元素类

4.1、收入条目,可以理解为一个对象的元素

 /**
* 收入账单
*/
public class IncomeBill implements IBill {
/**
* 收入明细金额
*/
private Double amount; /**
* 收入条目
*/
private String item; /**
* 收入明细金额
*/
public Double getAmount() {
return amount;
} /**
* 收入条目
*/
public String getItem() {
return item;
} public IncomeBill(Double amount, String item) {
this.amount = amount;
this.item = item;
} @Override
public void accept(IAccountBookViewer viewer) {
viewer.view(this);
}
}

IncomeBill.java

4.2、支出条目,可以理解为一个对象元素

 /**
* 消费账单
*/
public class ConsumeBill implements IBill { /**
* 支出明细金额
*/
private Double amount; /**
* 支出条目
*/
private String item; /**
* 支出明细金额
*/
public Double getAmount() {
return amount;
} /**
* 支出条目
*/
public String getItem() {
return item;
} public ConsumeBill(Double amount, String item) {
this.amount = amount;
this.item = item;
} @Override
public void accept(IAccountBookViewer viewer) {
viewer.view(this);
}
}

ConsumeBill.java

5、对象类

包含具体元素的集合,提供访问集合元素的轮询方法show

 import java.util.ArrayList;
import java.util.List; /**
* 账本类
*/
public class AccountBook {
/**
* 账单条目列表
*/
private List<IBill> billList = new ArrayList<IBill>(); /**
* 增加账单条目
* @param bill 账单条目
*/
public void addBill(IBill bill){
billList.add(bill);
} /**
* 访问者产看账本
* @param viewer
*/
public void show(IAccountBookViewer viewer){
for (IBill bill : billList) {
bill.accept(viewer);
}
} }

AccountBook.java

测试入口

 public class App {
public static void (String[] args) {
//创建账本
AccountBook accBook = new AccountBook(); //收入条目
accBook.addBill(new IncomeBill(10000d, "广告收入"));
accBook.addBill(new IncomeBill(900000d, "房地产项目")); //支出条目
accBook.addBill(new ConsumeBill(20000d, "工资"));
accBook.addBill(new ConsumeBill(10000d, "办公室租金"));
accBook.addBill(new ConsumeBill(8000d, "水电费")); //访问者
Boss boss = new Boss();
CPA cpa = new CPA(); //访问者查看账单
accBook.show(boss);
accBook.show(cpa); boss.getTotalConsume();
boss.getTotalIncome();
}
}

main方法

输出结果

 CPA查看所有收入是否已经缴税
CPA查看所有收入是否已经缴税
CPA查看是否工资已经扣税
老板查看总支出:38000.0
老板查看总收入:910000.0

示例源码:https://github.com/LF20160912/pattern

设计模式Design Pattern(4) -- 访问者模式的更多相关文章

  1. 设计模式Design Pattern(1)--简介

    什么是设计模式? 软件开发人员在长期实践中总结出来的解决特定问题的一套解决方案. 对象设计原则 计模式主要是基于以下的面向对象设计原则. 对接口编程而不是对实现编程. 优先使用对象组合而不是继承. 设 ...

  2. C#设计模式之二十一访问者模式(Visitor Pattern)【行为型】

    一.引言 今天我们开始讲“行为型”设计模式的第九个模式,该模式是[访问者模式],英文名称是:Visitor Pattern.如果按老规矩,先从名称上来看看这个模式,我根本不能获得任何对理解该模式有用的 ...

  3. 设计模式(Design Pattern)系列之.NET专题

    最近,不是特别忙,重新翻了下设计模式,特地在此记录一下.会不定期更新本系列专题文章. 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结. 使用 ...

  4. 设计模式23:Visitor 访问者模式(行为型模式)

    Visitor 访问者模式(行为型模式) 动机(Motivation)在软件构造过程中,由于需求的改变,某些类层次结构中常常需要增加新的行为(方法),如果直接在基类中做这样的修改,将会给子类带来繁重的 ...

  5. 设计模式-(13)访问者模式 (swift版)

    一,概念 访问者模式,是行为型设计模式之一.访问者模式是一种将数据操作与数据结构分离的设计模式,它可以算是 23 中设计模式中最复杂的一个,但它的使用频率并不是很高,大多数情况下,你并不需要使用访问者 ...

  6. 【design pattern】代理模式

    前言 设计模式分为三大类: 创建型模式:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式: 结构型模式:适配器模式.装饰器模式.代理模式.外观模式.桥接模式.组合模式.享元模式: 行为型模式 ...

  7. [Java复习] 设计模式 Design Pattern

    设计模式的六大原则 1.开闭原则(Open Close Principle) 对扩展开放,对修改关闭. 2.里氏代换原则(Liskov Substitution Principle) 任何基类可以出现 ...

  8. 【设计模式 - 24】之访问者模式(Visitor)

    1      模式简介 访问者模式的定义: 访问者模式将数据结构与数据操作进行了分离,解决了稳定的数据结构和易变的数据操作的耦合问题. 访问者模式的优点: 1)        符合单一职责原则: 2) ...

  9. 设计模式Design Pattern(3) -- 责任链模式

    什么是责任链模式? 责任链模式(Chain of Responsibility Pattern):请求知道公开接口,但不知道那个具体类处理,这些具体处理类对象连接成一条链.请求沿着这条链传递,直到有对 ...

随机推荐

  1. Python实现利用最大公约数求三个正整数的最小公倍数示例

    Python实现利用最大公约数求三个正整数的最小公倍数示例 本文实例讲述了Python实现利用最大公约数求三个正整数的最小公倍数.分享给大家供大家参考,具体如下: 在求解两个数的小公倍数的方法时,假设 ...

  2. mysql 各数据类型的大小及长度

    数字型 类型 大小 范围(有符号) 范围(无符号) 用途 TINYINT 1 字节 (-128,127) (0,255) 小整数值 SMALLINT 2 字节 (-32 768,32 767) (0, ...

  3. C#学习笔记三(委托·lambda表达式和事件,字符串和正则表达式,集合,特殊的集合)

    委托和事件的区别 序号 区别 委托 事件 1 是否可以使用=来赋值 是 否 2 是否可以在类外部进行调用 是 否 3 是否是一个类型 是 否,事件修饰的是一个对象 public delegate vo ...

  4. django.db.migrations.exceptions.BadMigrationError: Migration tests in app bl

    这个错误基本上都是 替换文件后才会出现的问题 因为你替换后他的日志文件没有完全替换的话,那么日志对应不到就会出现这样的问题, 一个模糊的处理办法:重新进行数据迁移:首先删除migrations中除去_ ...

  5. kh67-wjs

    个人简历 基本信息 姓    名:                           性    别: 年    龄:                                籍    贯: 联 ...

  6. PEP8-python编码规范(上)

    包含主要 Python 发行版中的标准库的 Python 代码的编码约定. 1.代码缩进 (1)每个缩进需要使用 4 个空格.一般使用一个Tab键. Python 3 不允许混合使用制表符和空格来缩进 ...

  7. 应用安全 - 社工 - By 大数据 - shodan - 汇总

    使用 | 命令 搜索语法 hostname: 搜索指定的主机或域名,例如 hostname:”google” port: 搜索指定的端口或服务,例如 port:”” country: 搜索指定的国家, ...

  8. xmake从入门到精通9:交叉编译详解

    xmake是一个基于Lua的轻量级现代化c/c 的项目构建工具,主要特点是:语法简单易上手,提供更加可读的项目维护,实现跨平台行为一致的构建体验. 除了win, linux, macOS平台,以及an ...

  9. 跨域设置之corsheaders

    安装 pip install django-cors-headers 注册应用 INSTALLED_APPS = ( ... 'corsheaders', ... ) 中间层设置 MIDDLEWARE ...

  10. setter 和 getter 高级 以及内存管理初级

    setter 和 getter 的演变,紧接setter 和 getter 初级 1.@property 和  @synthesize 这两个关键字的出现,就是为了剔除代码中的setter方法和get ...