访问者模式属于行为型模式;指将作用于某种数据结构中各元素的操作分离出来封装成独立的类,使其在不改变数据结构的前提下可以添加作用于这些元素的新的操作,为数据结构中的每个元素提供多种访问方式。它将对数据的操作与数据结构进行分离,是行为类模式中最复杂的一种模式。
访问者模式的目的是封装一些施加于某种数据结构元素之上的操作,一旦这些操作需要修改的话,接受这个操作的数据结构则可以保持不变。
数据结构的每一个节点都可以接受一个访问者的调用,此节点向访问者对象传入节点对象,而访问者对象则反过来执行节点对象的操作。这样的过程叫做“双重分派”。节点调用访问者,将它自己传入,访问者则将某算法针对此节点执行。
双重分派意味着施加于节点之上的操作是基于访问者和节点本身的数据类型,而不仅仅是其中的一者。
方法的接收者和方法的参数统称为方法的宗量。 根据分派基于宗量多少(接收者是一个宗量,参数是一个宗量),可以将分派分为单分派和多分派。单分派是指根据一个宗量就可以知道调用目标(即应该调用哪个方法),多分派需要根据多个宗量才能确定调用目标。
访问者模式的UML类图如下:

从上图可以看出,访问者模式涉及到抽象访问者角色、具体访问者角色、抽象节点角色、具体节点角色、结构对象角色以及客户端角色等6个角色:
从上图可以看到,抽象访问者角色为每一个具体节点都准备了一个访问操作,由于有两个节点,因此,对应的就有两个访问操作。
我们都看过很多歌唱选秀类的综艺节目,当某个表演者表演完毕后,下面的评委需要对其表演进行评价(晋级,淘汰,待定),决定其是否晋级下一轮,我们这里就以这个为例子讲解访问者设计模式。
例子的UML类图:

抽象节点角色:
package com.charon.visitor;
/**
* @className: Person
* @description:
* @author: charon
* @create: 2022-03-27 17:18
*/
public abstract class Person {
abstract void accept(Action action);
}
具体节点角色:
package com.charon.visitor;
/**
* @className: Man
* @description:
* @author: charon
* @create: 2022-03-27 17:19
*/
public class Man extends Person {
private String name;
public Man(String name) {
this.name = name;
}
/**
* Gets the value of name
*
* @return the value of name
*/
public String getName() {
return name;
}
@Override
void accept(Action action) {
action.getManResult(this);
}
}
package com.charon.visitor;
/**
* @className: Woman
* @description:
* @author: charon
* @create: 2022-03-27 17:19
*/
public class Woman extends Person{
private String name;
public Woman(String name) {
this.name = name;
}
/**
* Gets the value of name
*
* @return the value of name
*/
public String getName() {
return name;
}
@Override
void accept(Action action) {
action.getWomanResult(this);
}
}
抽象访问者角色:
package com.charon.visitor;
/**
* @className: Action
* @description:
* @author: charon
* @create: 2022-03-27 17:19
*/
public abstract class Action {
/**
* 得到男人的评价结果
* @param man
*/
abstract void getManResult(Man man);
/**
* 得到女人的评价结果
* @param woman
*/
abstract void getWomanResult(Woman woman);
}
具体访问者角色:
package com.charon.visitor;
/**
* @className: FailAction
* @description:
* @author: charon
* @create: 2022-03-27 17:21
*/
public class FailAction extends Action{
@Override
void getManResult(Man man) {
System.out.println(man.getName() + " 给的评价是该表演者表演淘汰。。。。");
}
@Override
void getWomanResult(Woman woman) {
System.out.println(woman.getName() + " 给的评价是该表演者表演淘汰。。。。");
}
}
package com.charon.visitor;
/**
* @className: SuccessAction
* @description:
* @author: charon
* @create: 2022-03-27 17:22
*/
public class SuccessAction extends Action{
@Override
void getManResult(Man man) {
System.out.println(man.getName() + " 给的评价是该表演者表演晋级。。。。");
}
@Override
void getWomanResult(Woman woman) {
System.out.println(woman.getName() + " 给的评价是该表演者表演晋级。。。。");
}
}
结构对象角色:
package com.charon.visitor;
import java.util.ArrayList;
import java.util.List;
/**
* @className: ObjectStructure
* @description:
* @author: charon
* @create: 2022-03-27 17:25
*/
public class ObjectStructure {
/**
* 集合,用于存放表演的评价者
*/
private List<Person> persons = new ArrayList<>();
public void add(Person person){
persons.add(person);
}
public void remove(Person person){
persons.remove(person);
}
/**
* 显示评价结果
* @param action
*/
public void display(Action action){
for (Person person : persons) {
person.accept(action);
}
}
}
测试:
package com.charon.visitor;
/**
* @className: Client
* @description:
* @author: charon
* @create: 2022-03-27 17:29
*/
public class Client {
public static void main(String[] args) {
ObjectStructure structure = new ObjectStructure();
structure.add(new Man("男评委1"));
structure.add(new Man("男评委2"));
structure.add(new Woman("女评委1"));
structure.add(new Woman("女评委2"));
structure.display(new SuccessAction());
}
}
打印:
男评委1 给的评价是该表演者表演晋级。。。。
男评委2 给的评价是该表演者表演晋级。。。。
女评委1 给的评价是该表演者表演晋级。。。。
女评委2 给的评价是该表演者表演晋级。。。。
如上所示,访问者模式的代码就完成了,如果现在需要添加一个待定的操作类型,就只需要添加一个WaitAction就行了:
package com.charon.visitor;
/**
* @className: WaitAction
* @description:
* @author: charon
* @create: 2022-03-27 17:39
*/
public class WaitAction extends Action{
@Override
void getManResult(Man man) {
System.out.println(man.getName() + " 给的评价是该表演者表演待定。。。。");
}
@Override
void getWomanResult(Woman woman) {
System.out.println(woman.getName() + " 给的评价是该表演者表演待定。。。。");
}
}
访问者模式的优点如下:
访问者模式的缺点如下:
由于访问者模式的这些缺点,导致很多人反对使用访问者模式。
当系统中存在类型数量稳定(固定)的一类数据结构时,可以使用访问者模式方便地实现对该类型所有数据结构的不同操作,而又不会对数据产生任何副作用(脏数据)。简而言之,就是当对集合中的不同类型数据(类型数量稳定)进行多种操作时,使用访问者模式。
通常在以下情况可以考虑使用访问者模式。
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
我有一个包含模块的模型。我想在模块中覆盖模型的访问器方法。例如:classBlah这显然行不通。有什么想法可以实现吗? 最佳答案 您的代码看起来是正确的。我们正在毫无困难地使用这个确切的模式。如果我没记错的话,Rails使用#method_missing作为属性setter,因此您的模块将优先,阻止ActiveRecord的setter。如果您正在使用ActiveSupport::Concern(参见thisblogpost),那么您的实例方法需要进入一个特殊的模块:classBlah
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
我正在使用Sequel构建一个愿望list系统。我有一个wishlists和itemstable和一个items_wishlists连接表(该名称是续集选择的名称)。items_wishlists表还有一个用于facebookid的额外列(因此我可以存储opengraph操作),这是一个NOTNULL列。我还有Wishlist和Item具有续集many_to_many关联的模型已建立。Wishlist类也有:selectmany_to_many关联的选项设置为select:[:items.*,:items_wishlists__facebook_action_id].有没有一种方法可以
我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数
给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最
目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称
项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU