Java描述设计模式(23):访问者模式
本文源码:GitHub·点这里 || GitEE·点这里
一、生活场景
1、场景描述
电竞是游戏比赛达到“竞技”层面的体育项目。利用电子设备作为运动器械进行的、人与人之间的智力对抗运动。通过电竞,可以提高人的反应能力、协调能力、团队精神等。但是不同人群的对电竞的持有的观念不一样,有的人认为电竞就是沉迷网络,持反对态度,而有的人就比较赞同。下面基于访问者模式来描述该场景。
2、场景图解
3、代码实现
public class C01_InScene { public static void main(String[] args) { DataSet dataSet = new DataSet() ; dataSet.addCrowd(new Youth()); dataSet.addCrowd(new MiddleAge()); CrowdView crowdView = new Against() ; dataSet.display(crowdView); crowdView = new Approve() ; dataSet.display(crowdView); } } /** * 双分派,不同人群管理 */ abstract class Crowd { abstract void accept(CrowdView action); } class Youth extends Crowd { @Override public void accept(CrowdView view) { view.getYouthView(this); } } class MiddleAge extends Crowd { @Override public void accept(CrowdView view) { view.getMiddleAgeView (this); } } /** * 不同人群观念的管理 */ abstract class CrowdView { // 青年人观念 abstract void getYouthView (Youth youth); // 中年人观念 abstract void getMiddleAgeView (MiddleAge middleAge); } class Approve extends CrowdView { @Override public void getYouthView(Youth youth) { System.out.println("青年人赞同电竞"); } @Override public void getMiddleAgeView(MiddleAge middleAge) { System.out.println("中年人赞同电竞"); } } class Against extends CrowdView { @Override public void getYouthView(Youth youth) { System.out.println("青年人反对电竞"); } @Override public void getMiddleAgeView(MiddleAge middleAge) { System.out.println("中年人反对电竞"); } } /** * 提供一个数据集合 */ class DataSet { private List<Crowd> crowdList = new ArrayList<>(); public void addCrowd (Crowd crowd) { crowdList.add(crowd); } public void display(CrowdView crowdView) { for(Crowd crowd : crowdList) { crowd.accept(crowdView); } } }
二、访问者模式
1、基础概念
访问者模式是对象的行为模式,把作用于数据结构的各元素的操作封装,操作之间没有关联。可以在不改变数据结构的前提下定义作用于这些元素的不同的操作。主要将数据结构与数据操作分离,解决数据结构和操作耦合问题核心原理:被访问的类里面加对外提供接待访问者的接口。
2、模式图解
3、核心角色
- 抽象访问者角色
声明多个方法操作,具体访问者角色需要实现的接口。
- 具体访问者角色
实现抽象访问者所声明的接口,就是各个访问操作。
- 抽象节点角色
声明接受操作,接受访问者对象作为参数。
- 具体节点角色
实现抽象节点所规定的具体操作。
- 结构对象角色
能枚举结构中的所有元素,可以提供一个高层的接口,用来允许访问者对象访问每一个元素。
4、源码实现
public class C02_Visitor { public static void main(String[] args) { ObjectStructure obs = new ObjectStructure(); obs.add(new NodeA()); obs.add(new NodeB()); Visitor visitor = new VisitorA(); obs.doAccept(visitor); } } /** * 抽象访问者角色 */ interface Visitor { /** * NodeA的访问操作 */ void visit(NodeA node); /** * NodeB的访问操作 */ void visit(NodeB node); } /** * 具体访问者角色 */ class VisitorA implements Visitor { @Override public void visit(NodeA node) { node.operationA() ; } @Override public void visit(NodeB node) { node.operationB() ; } } class VisitorB implements Visitor { @Override public void visit(NodeA node) { node.operationA() ; } @Override public void visit(NodeB node) { node.operationB() ; } } /** * 抽象节点角色 */ abstract class Node { /** * 接收访问者 */ abstract void accept(Visitor visitor); } /** * 具体节点角色 */ class NodeA extends Node{ @Override public void accept(Visitor visitor) { visitor.visit(this); } public void operationA(){ System.out.println("NodeA.operationA"); } } class NodeB extends Node{ @Override public void accept(Visitor visitor) { visitor.visit(this); } public void operationB(){ System.out.println("NodeB.operationB"); } } /** * 结构对象角色类 */ class ObjectStructure { private List<Node> nodes = new ArrayList<>(); public void detach(Node node) { nodes.remove(node); } public void add(Node node){ nodes.add(node); } public void doAccept(Visitor visitor){ for(Node node : nodes) { node.accept(visitor); } } }
三、Spring框架应用
1、Bean结构的访问
BeanDefinitionVisitor类,遍历bean的各个属性;接口 BeanDefinition,定义Bean的各样信息,比如属性值、构造方法、参数等等。这里封装操作bean结构的相关方法,但却没有改变bean的结构。
2、核心代码块
public class BeanDefinitionVisitor { public void visitBeanDefinition(BeanDefinition beanDefinition) { this.visitParentName(beanDefinition); this.visitBeanClassName(beanDefinition); this.visitFactoryBeanName(beanDefinition); this.visitFactoryMethodName(beanDefinition); this.visitScope(beanDefinition); if (beanDefinition.hasPropertyValues()) { this.visitPropertyValues(beanDefinition.getPropertyValues()); } if (beanDefinition.hasConstructorArgumentValues()) { ConstructorArgumentValues cas = beanDefinition.getConstructorArgumentValues(); this.visitIndexedArgumentValues(cas.getIndexedArgumentValues()); this.visitGenericArgumentValues(cas.getGenericArgumentValues()); } } }
四、模式总结
1、优点描述
(1) 访问者模式符合单一职责原则、使程序具有良好的扩展性、灵活性;
(2) 访问者模式适用与拦截器与过滤器等常见功能,数据结构相对稳定的场景;
2、缺点描述
(1) 访问者关注其他类的内部细节,依赖性强,违反迪米特法则,这样导致具体元素更新麻烦;
(2) 访问者依赖具体元素,不是抽象元素,面向细节编程,违背依赖倒转原则;
五、源代码地址
GitHub·地址 https://github.com/cicadasmile/model-arithmetic-parent GitEE·地址 https://gitee.com/cicadasmile/model-arithmetic-parent
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
面试还搞不懂redis,快看看这40道面试题(含答案和思维导图)
Redis 面试题 1、什么是 Redis?.2、Redis 的数据类型?3、使用 Redis 有哪些好处?4、Redis 相比 Memcached 有哪些优势?5、Memcache 与 Redis 的区别都有哪些?6、Redis 是单进程单线程的?7、一个字符串类型的值能存储最大容量是多少?8、Redis 的持久化机制是什么?各自的优缺点?9、Redis 常见性能问题和解决方案:10、redis 过期键的删除策略?11、Redis 的回收策略(淘汰策略)?12、为什么 edis 需要把所有数据放到内存中?13、Redis 的同步机制了解么?14、Pipeline 有什么好处,为什么要用 pipeline?15、是否使用过 Redis 集群,集群的原理是什么?16、Redis 集群方案什么情况下会导致整个集群不可用?17、Redis 支持的 Java 客户端都有哪些?官方推荐用哪个?18、Jedis 与 Redisson 对比有什么优缺点?19、Redis 如何设置密码及验证密码?20、说说 Redis 哈希槽的概念?21、Redis 集群的主从复制模型是怎样的?22、Redis 集群...
- 下一篇
spring-boot-starter-pay 0.0.3 发布,一行代码发起支付与配置
spring-boot-starter-pay是一个基于spring-boot实现自动化配置的支付对接工具包,让你真正做到一行代码实现支付聚合,让你可以不用理解支付怎么对接,只需要专注你的业务。 优雅的轻量级支付模块集成支付对接支付整合(微信、支付宝、银联、友店、富友、跨境支付paypal、payoneer(P卡派安盈)易极付)app、扫码、网页支付刷卡付条码付刷脸付转账服务商模式、支持多种支付类型多支付账户、支付与业务完全剥离,简单几行代码即可实现支付,简单快速完成支付模块的开发。 软件框架依赖 spring-boot pay-java-parent 0.0.3 更新内容如下: 1.支付消息回调处理器外部配置化 2.支付消息回调拦截器外部配置化 3.证书获取接口话并提供多种证书配置的获取方式 4.暴露支付回调参数对象化获取 5.刷脸支付
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- 设置Eclipse缩进为4个空格,增强代码规范
- CentOS8安装Docker,最新的服务器搭配容器使用
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- CentOS8编译安装MySQL8.0.19
- SpringBoot2整合Redis,开启缓存,提高访问速度
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Hadoop3单机部署,实现最简伪集群
- CentOS7,CentOS8安装Elasticsearch6.8.6