访问者模式

一、概述

一般问题:对象结构比较稳定,但经常需要在此对象结构上定义新的操作。

核心方案:封装一些作用于某种数据结构中的各元素的操作,它可以在不改变这个数据结构的前提下定义作用于这些元素的新的操作。

设计意图:一般情况下,一个类会封装一些属性,同时会提供访问这些属性的方法。如果这个类的属性是稳定的,但是访问属性的接口却经常变动,也就是经常需要修改类代码,显然不符合“开闭原则”。在设计模式里,没有什么是不能拆的,既然数据结构稳定,数据操作经常改变,那就把数据操作封装到单独的类中好了,然后再在这些类中定义好对每种数据类型的访问方法,这就是最初的访问者模式了。

 

访问者模式的类图在众多设计模式里属于比较复杂的。我们分成两部分来看:

(1)数据结构端

  1. Element定义元素类,是所有数据类型的抽象类,需要定义accept接口,用来接受访问者。
  2. ObjectStructure为元素类容器

(2)数据访问端

Visitor定义访问者类,是所有访问者的抽象类,需要定义各个元素的visite接口,用于对不同元素的区别对待


 二、应用实战

 实际开发中用到访问者模式的地方很少,没有必要无中生有一个故事,硬套上设计模式。Java新特性——注解——其解析原理的确用到了访问者模式。编译时Annotation解析的基本原理是:在某些代码元素上(如类型,函数,字段等)添加注解,在编译时编译器会检查AbstractProcessor的子类,并且调用该类型的process函数,然后将添加了注解的所有元素都传递到process函数中,使得开发人员可以在编译期进行相应的处理, 例如根据注解生成新的java类,这也就是ButterKnife等开源库的基本原理.

注解的对象可以是以下类型:
  • PackageElement - 包元素,包含某个包下面的信息,可以获取到包名等
  • TypeElement - 类型元素,如某个字段属于某种类型
  • ExecutableElement - 可执行元素,代表了函数类型的元素
  • VariableElement - 变量元素
  • TypeParameterElement - 类型参数元素
注解的写法:
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation {
    //定义注解的属性,这不是方法
    String name();//必选注解
    int value() default 20;//有属性就是可选属性
}

Element类如下:

public interface Element {
    //代码省略
    //获取元素类型
    ElementKind getKind();
    
    //获取修饰符,如public 、 static 、final等
    Set<Modifier> getModifiers();
 
    //接受访问者的访问
    <R, P> R accept(ElementVisitor<R, P> P p);
}

Element中有一个<R, P> R accept(ElementVisitor<R, P> var1, P var2); ,这里就是利用访问者模式去访问Element中的所有元素信息。

再看Visitor类:

public interface ElementVisitor<R, P> {
 
    //访问元素
    R visit(Element e, P p);
 
    //访问包元素
    R visitPackage(PackageElement e, P p);
 
    //访问一个类型元素
    R visitType(TypeElement e, P p);
 
    //访问一个变量类型
    R visitVariable(VariableElement e, P p);
 
    //访问一个可执行元素
    R visitExecutable(ExecutableElement e, P p);
 
    //访问一个参数元素
    R visitTypeParameter(TypeParameterElement e, P p);
 
    //处理未知的元素类型,这是为了应对后续java语言的扩展而预留的接口
    //例如后续元素类型增加,那么通过这个接口就可以处理上述没有声明的类型
    R visitUnknown(Element e, P p);
}

ElementVisitor中定义了很多visit方法,每个visit方法对应一种元素,这就是典型的访问者模式。一个类元素和一个函数元素是完全不一样的,它们的结构不一样,因此,编译器对它们的操作也不同,通过访问者模式正好解决数据结构与数据操作分离的问题。


三、总结

总结:访问者模式是一种行为型设计模式,在现实情况下,我们根据具体的情况来评估是否适合使用访问者模式,例如,我们的对象结构是否稳定,使用访问者模式是否优化我们的代码,而不是使得代码变得更加复杂。

用一句话来表述访问者模式:

横看成岭侧成峰

优点:使得数据结构和作用于结构上的操作解耦,使得操作集合可以独立变化。

缺点:

  • 具体元素变更时导致修改成本大
  • 违反了依赖倒置原则,为了达到区别对待而依赖了具体类,没有依赖抽象
 

转载于:https://www.cnblogs.com/not2/p/11089979.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
访问者模式是GOF设计模式中的一种行为型模式。它被描述为最复杂且最难以理解的一种模式。访问者模式的目的是封装一些作用于某种数据结构中的各元素的操作,可以在不改变这些元素的类的前提下定义这些操作。它通过将数据结构与操作分离,使得可以在不改变数据结构的前提下添加新的操作,提高了代码的灵活性和可扩展性。 访问者模式的构成包括元素(Element)、访问者(Visitor)和对象结构(Object Structure)。元素表示数据结构中的各个元素,它们通常会提供一个接受访问者的方法。访问者则表示对元素的操作,它们可以根据具体的元素类型进行不同的操作。对象结构则是一个容器,用于存放元素,并提供让访问者访问元素的接口。 访问者模式适用于数据结构相对稳定,但其操作经常发生变化的情况。它可以将操作的变化封装在访问者中,而无需改变元素的类。这样一来,当需要新增一种操作时,只需要新增一个访问者,而不需要改变元素的类。 访问者模式的优点包括增加新的操作非常方便,由于具体访问者类与具体元素类之间没有直接的关联,因此增加新的访问者类对原有类库无影响。同时,它也符合开闭原则,对于元素的类库可以在不修改源代码的情况下添加新的操作。 总结起来,访问者模式是一种通过将数据结构与操作分离的设计模式,可以在不改变数据结构的前提下添加新的操作。它的构成包括元素、访问者和对象结构,适用于数据结构相对稳定,但操作经常变化的情况。访问者模式的优点包括增加新的操作方便,符合开闭原则。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [详解设计模式:访问者模式](https://blog.csdn.net/weixin_45187434/article/details/128197861)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [设计模式 访问者模式](https://download.csdn.net/download/zjn640322/9685149)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值