设计模式 -- 访问者模式(Visitor Pattern)

访问者模式是一种设计模式,用于封装作用于特定数据结构的操作。它在JDK和Spring源码中有应用,如FileVisitor接口用于遍历文件树,Spring的BeanDefinitionVisitor则用于访问Bean定义信息。这种模式在数据结构稳定而操作频繁变化的系统中能够增加新的操作而不会影响原有结构。然而,增加新的数据结构元素会变得困难,可能导致系统复杂性增加。
摘要由CSDN通过智能技术生成

封装一些作用于某些数据结构中的各元素的操作,它可以在不改变数据结构的前提下赋予这些元素新的操作。

应用场景

  • 对象结构比较稳定,但是需要在对象结构的基础上定义新的操作。

  • 需要对同一个类的不同对象执行不不同的操作,但是不希望增加操作的时候改变这些类。

总结

  • 准确识别出Visitor实用的场景,如果一个对象结构不稳定决不可使用,不然在增删元素时改动将非常巨大。

  • 对象结构中的元素要可以迭代访问

  • Visitor里一般存在与元素个数相同的visit方法。

  • 元素通过 this 方法通过 accept 将自己传递给了Visitor。

特点:

  • 访问者模式把数据结构和作用于结构上的操作解耦合,使得操作集合可相对自由地演化。

  • 访问者模式适用于数据结构相对稳定算法又易变化的系统。因为访问者模式使得算法操作增加变得容易。若系统数据结构对象易于变化,经常有新的数据对象增加进来,则不适合使用访问者模式。

  • 访问者模式的优点是增加操作很容易,因为增加操作意味着增加新的访问者。访问者模式将有关行为集中到一个访问者对象中,其改变不影响系统数据结构。其缺点就是增加新的数据结构很困难。

优点:

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

  • 添加新的操作或者说访问者会非常容易。

  • 将对各个元素的一组操作集中在一个访问者类当中。

  • 使得类层次结构不改变的情况下,可以针对各个层次做出不同的操作,而不影响类层次结构的完整性。

  • 可以跨越类层次结构,访问不同层次的元素类,做出相应的操作。

  • 使得给结构稳定的对象增加新算法变得容易,提搞了代码的可维护性,可扩展性。

缺点:

  • 增加新的元素会非常困难。

  •  实现起来比较复杂,会增加系统的复杂性。

  • 破坏封装,如果将访问行为放在各个元素中,则可以不暴露元素的内部结构和状态,但使用访问者模式的时候,为了让访问者能获取到所关心的信息,元素类不得不暴露出一些内部的状态和结构,就像收入和支出类必须提供访问金额和单子的项目的方法一样。

适用性:

  • 数据结构稳定,作用于数据结构的操作经常变化的时候。

  • 当一个数据结构中,一些元素类需要负责与其不相关的操作的时候,为了将这些操作分离出去,以减少这些元素类的职责时,可以使用访问者模式。

  • 有时在对数据结构上的元素进行操作的时候,需要区分具体的类型,这时使用访问者模式可以针对不同的类型,在访问者类中定义不同的操作,从而去除掉类型判断。

举例是这样,但是不合理:

应该是:

public class VisitorPattern {
    public static void main(String[] args) {
        ObjectStructure os = new ObjectStructure();
        os.add(new ConcreteElementA());
        os.add(new ConcreteElementB());
        Visitor visitor = new ConcreteVisitorA();
        os.accept(visitor);
        System.out.println("------------------------");
        visitor = new ConcreteVisitorB();
        os.accept(visitor);
    }
}


//抽象访问者
interface Visitor {
    void visit(ConcreteElementA element);

    void visit(ConcreteElementB element);
}


//具体访问者A类
class ConcreteVisitorA implements Visitor {
    public void visit(ConcreteElementA element) {
        System.out.println("具体访问者A访问-->" + element.operationA());
    }

    public void visit(ConcreteElementB element) {
        System.out.println("具体访问者A访问-->" + element.operationB());
    }
}


//具体访问者B类
class ConcreteVisitorB implements Visitor {
    public void visit(ConcreteElementA element) {
        System.out.println("具体访问者B访问-->" + element.operationA());
    }

    public void visit(ConcreteElementB element) {
        System.out.println("
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值