什么是访问者模式?

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

解决的主要问题

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

如何实现

(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. Closure - Mimicking block scope

    The basic syntax of an anoymous function used as a block scope (often called a private scope) is as ...

  2. [转] javascript 正则表达式提取数字使用

    简述: 测试了一下js的正则表达式函数match 用来从一个字符串里挑出非0开头的数字, 放入一个array里, 之后join(',')之后输出 知识点: 1. 用match函数, 提取一个字符串当中 ...

  3. mysql用sql语句创建表和数据 设置字符编码为utf-8

    简而言之 CREATE DATABASE xx CHARACTER SET utf8 COLLATE utf8_general_ci; USE xx; ),qname ) ) ) ) )); ) ,t ...

  4. 双系统正确卸载Ubuntu系统

    双系统正确卸载Ubuntu系统  安装系统后由于显卡驱动问题,无法开机,从而只能卸载重装,重装过程如下. 第一步:下载需要的工具包,这里我用的是MBRfix, 可以直接从我分享的网盘链接下载,密码gw ...

  5. 【Qt开发】Win7 64位qt-windows-x86-msvc2015-5.6.0 DLL依赖库打包

    Win7 64位qt-windows-x86-msvc2015-5.6.0 DLL依赖库打包 今天开始系统的学习QT,第一个测试的问题就是在纯净的系统中如何正常运行,也就是找出QT生成的exe的依赖库 ...

  6. 【VS开发】【智能语音处理】MATLAB 与 音频处理 相关内容摘记

    MATLAB 与 音频处理 相关内容摘记 MATLAB 与 音频处理 相关内容摘记 1 MATLAB 音频相关函数 1 MATLAB 处理音频信号的流程 2 音量标准化 2 声道分离合并与组合 3 数 ...

  7. 【VS开发】【图像处理】自动白平衡(AWB)算法---色温曲线

    原文地址:http://blog.csdn.net/wzwxiaozheng/article/details/38434391 白平衡算法---色温曲线 本文大体讲解了白平衡的算法流程,适用于想了解和 ...

  8. Java 并发编程:核心理论(一)

    前言......... 并发编程是Java程序员最重要的技能之一,也是最难掌握的一种技能.它要求编程者对计算机最底层的运作原理有深刻的理解,同时要求编程者逻辑清晰.思维缜密,这样才能写出高效.安全.可 ...

  9. Mysql-慢查询排查

    1.开启慢日志 2.使用show full processlist抓出慢查询语句 3.使用explain分析语句 4. set global profiling = ON

  10. Educational Codeforces Round 64 -B(贪心)

    题目链接:https://codeforces.com/contest/1156/problem/B 题意:给一段字符串,通过变换顺序使得该字符串不包含为位置上相邻且在字母表上也相邻的情况,并输出. ...