1. 设计模式之访问者模式:动态的给目标对象增加新功能

1.1. 介绍

在访问者模式(Visitor Pattern)中,我们使用了一个访问者类,它改变了元素类的执行算法。

通过这种方式,元素的执行算法可以随着访问者改变而改变。这种类型的设计模式属于行为型模式。根据模式,元素对象已接受访问者对象,这样访问者对象就可以处理元素对象上的操作。

访问者模式,一般,就是跟组合模式结合起来使用的,组合模式代表了一种复杂的对象的类型,如果后面要给树形的数据结构增加什么功能,修改代码可能会比较麻烦,但是如果采用访问者模式,可以在 任何时候 给树形的数据结构 增加任何的功能

应用场景

当需要对一个对象结构中的对象执行多种不同的且不相关的操作时,尤其是这些操作需要避免"污染"对象类本身。

优缺点及建议

优点
1.单一职责原则:访问者模式符合单一职责原则,每个类只负责一项职责。
2.扩展性:容易为数据结构添加新的操作。
3.灵活性:访问者可以独立于数据结构变化。

缺点
1.增加新元素类型困难(需修改所有访问者)。
2.破坏封装(访问者需访问元素内部细节)。
3.复杂对象结构可能导致访问者类膨胀。

使用建议
1.当对象结构稳定,但需要在其上定义多种新操作时,考虑使用访问者模式。
2.当需要避免操作"污染"对象类时,使用访问者模式封装操作。

注意事项
1.若元素类型频繁变化,避免使用访问者模式。

1.2. 实现及相关代码

假设有一个部门业务,数据结构是树形结构,要对数据进行一些操作。

1.2.1. 相关代码

接口及数据对象

public static class Department {
    private String name;
    private List<Department> children = new ArrayList<Department>();

    public Department(String name) {
        super();
        this.name = name;
    }

    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public List<Department> getChildren() {
        return children;
    }
    public void setChildren(List<Department> children) {
        this.children = children;
    }

    public void accept(Visitor visitor) {
        visitor.visit(this);
    }
}

public interface Visitor {
    void visit(Department dept);
}

public static class RemoveVisitor implements Visitor {
    public void visit(Department dept) {
        if(dept.getChildren().size() > 0) {
            for(Department child : dept.getChildren()) {
                child.accept(this);
            }
        }
        System.out.println("删除部门【" + dept.getName() + "】");
    }
}

public static class UpdateStatusVisitor implements Visitor {
    private String status;

    public void visit(Department dept) {
        if(dept.getChildren().size() > 0) {
            for(Department child : dept.getChildren()) {
                child.accept(this);
            }
        }
        System.out.println("将部门【" + dept.getName() + "】的状态修改为:" + status);
    }

    public UpdateStatusVisitor(String status) {
        this.status = status;
    }
}

调用方法

public static void main(String[] args) {
    Department leafDept1 = new Department("叶子部门1");
    Department leafDept2 = new Department("叶子部门2");
    Department leafDept3 = new Department("叶子部门3");

    Department subDept1 = new Department("子部门1");
    subDept1.getChildren().add(leafDept1);
    subDept1.getChildren().add(leafDept2);

    Department subDept2 = new Department("子部门2");
    subDept2.getChildren().add(leafDept3);

    Department parentDept = new Department("父部门");
    parentDept.getChildren().add(subDept1);
    parentDept.getChildren().add(subDept2);

    Visitor removeVisitor = new RemoveVisitor();
    parentDept.accept(removeVisitor);

    Visitor updateStatusVisitor = new UpdateStatusVisitor("禁用");
    parentDept.accept(updateStatusVisitor);
}

results matching ""

    No results matching ""