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);
}