设计模式Design Pattern(4) -- 访问者模式
什么是访问者模式?
一个对象有稳定的数据结构,却为不同的访问者提供不同的数据操作,对象提供接收访问者的方法,从而保证数据结构的稳定性和操作的多样性。也可以理解为,封装对象的操作方法,达到不改变对象数据结构的稳定性同时易于扩展操作。
解决的主要问题
主要解决:稳定的数据结构和易变的操作耦合问题。
如何实现
(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) -- 访问者模式的更多相关文章
- 设计模式Design Pattern(1)--简介
什么是设计模式? 软件开发人员在长期实践中总结出来的解决特定问题的一套解决方案. 对象设计原则 计模式主要是基于以下的面向对象设计原则. 对接口编程而不是对实现编程. 优先使用对象组合而不是继承. 设 ...
- C#设计模式之二十一访问者模式(Visitor Pattern)【行为型】
一.引言 今天我们开始讲“行为型”设计模式的第九个模式,该模式是[访问者模式],英文名称是:Visitor Pattern.如果按老规矩,先从名称上来看看这个模式,我根本不能获得任何对理解该模式有用的 ...
- 设计模式(Design Pattern)系列之.NET专题
最近,不是特别忙,重新翻了下设计模式,特地在此记录一下.会不定期更新本系列专题文章. 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结. 使用 ...
- 设计模式23:Visitor 访问者模式(行为型模式)
Visitor 访问者模式(行为型模式) 动机(Motivation)在软件构造过程中,由于需求的改变,某些类层次结构中常常需要增加新的行为(方法),如果直接在基类中做这样的修改,将会给子类带来繁重的 ...
- 设计模式-(13)访问者模式 (swift版)
一,概念 访问者模式,是行为型设计模式之一.访问者模式是一种将数据操作与数据结构分离的设计模式,它可以算是 23 中设计模式中最复杂的一个,但它的使用频率并不是很高,大多数情况下,你并不需要使用访问者 ...
- 【design pattern】代理模式
前言 设计模式分为三大类: 创建型模式:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式: 结构型模式:适配器模式.装饰器模式.代理模式.外观模式.桥接模式.组合模式.享元模式: 行为型模式 ...
- [Java复习] 设计模式 Design Pattern
设计模式的六大原则 1.开闭原则(Open Close Principle) 对扩展开放,对修改关闭. 2.里氏代换原则(Liskov Substitution Principle) 任何基类可以出现 ...
- 【设计模式 - 24】之访问者模式(Visitor)
1 模式简介 访问者模式的定义: 访问者模式将数据结构与数据操作进行了分离,解决了稳定的数据结构和易变的数据操作的耦合问题. 访问者模式的优点: 1) 符合单一职责原则: 2) ...
- 设计模式Design Pattern(3) -- 责任链模式
什么是责任链模式? 责任链模式(Chain of Responsibility Pattern):请求知道公开接口,但不知道那个具体类处理,这些具体处理类对象连接成一条链.请求沿着这条链传递,直到有对 ...
随机推荐
- vue-cli本地环境API代理设置和解决跨域
前言 我们在使用vue-cli启动项目的时候npm run dev便可以启动我们的项目了,通常我们的请求地址是以localhost:8080来请求接口数据的,localhost是没有办法设置cooki ...
- Day06:方法 / 猜字母游戏
JAVA方法 方法就是处理一个业务所需要编写的代码的代码段 方法特性 一个方法处理一个业务 方法代码编写,不和其他方法冲突 方法定义后可以随意调用 将main方法中的所有代码分散到各个普通方法中 减少 ...
- XSS-反射型
前情提要:html的dom对象:document 如document.cookie / document.write() http://netsecurity.51cto.com/art/20131 ...
- WorkStation Linux 客户端 虚拟机的使用过程
1. 安装Workstation 版本选择 12.5 以上的版本 2. 安装完成之后 选择 打开虚拟机->选择 ovf文件 选择 ovf 文件 选择读入即可 3. 设置虚拟机的选项: 3.1 ...
- PHP手册在7.1迁移页面给出了替代方案,就是用OpenSSL取代MCrypt.
/** * [AesSecurity aes加密,支持PHP7.1] */ class AesSecurity { /** * [encrypt aes加密] * @p ...
- phpstorm配合xdebug进行本地调试代码
笔者在使用的环境是wamp3.1.6和phpstorm2018 ,php选择的环境是php7.2 1. 在php.ini中添加xdebug的配置信息 首先建议是先找对php.ini的位置,可以在php ...
- emmet语法列表
emmet语法 来源:https://docs.emmet.io/cheat-sheet/ Child: > nav>ul>li <nav> <ul> &l ...
- python-2:爬取某个网页(虎扑)帖子的标题做词云图
关键词:requests,BeautifulSoup,jieba,wordcloud 整体思路:通过requests请求获得html,然后BeautifulSoup解析html获得一些关键数据,之后通 ...
- Nginx教程(一)-全面认知
什么是 nginx nginx 是一款高性能的 http 服务器,反向代理服务器,电子邮件(IMAP/POP3)代理服务器: 它的特点就是高性能,占用内存少,支持高并发,运行稳定: 官方测试 可支持 ...
- 在 Chrome DevTools 中调试 JavaScript 入门
第 1 步:重现错误 找到一系列可一致重现错误的操作始终是调试的第一步. 点击 Open Demo. 演示页面随即在新标签中打开. OPEN DEMO 在 Number 1 文本框中输入 5. 在 N ...