访问者模式核心思想
访问者模式(Visitor Pattern)的核心在于将数据结构与数据操作分离。当系统拥有一个相对稳定的对象结构,但需要频繁添加新操作时,访问者模式通过双重分派(Double Dispatch)机制,使得操作能够灵活地添加到对象结构上,而无需修改结构中的各个类。
双重分派机制揭秘
在Java中,方法调用依赖于对象的运行时类型,但方法重载则依赖于参数的编译时类型。访问者模式通过两次方法调用来实现双重分派:
// 第一次分派:accept方法根据元素实际类型调用具体元素方法
element.accept(visitor);
// 第二次分派:visit方法根据访问者实际类型调用具体访问者方法
visitor.visit(this);
这种机制使得访问者能够根据元素的具体类型执行相应的操作,实现了运行时动态绑定。
实战示例:文档处理系统
假设我们有一个文档处理系统,包含多种元素类型:
// 定义文档元素接口
interface DocumentElement {
void accept(DocumentVisitor visitor);
}
// 具体元素:文本段落
class TextElement implements DocumentElement {
public void accept(DocumentVisitor visitor) {
visitor.visit(this);
}
}
// 具体元素:图像元素
class ImageElement implements DocumentElement {
public void accept(DocumentVisitor visitor) {
visitor.visit(this);
}
}
// 定义访问者接口
interface DocumentVisitor {
void visit(TextElement text);
void visit(ImageElement image);
}
// 具体访问者:导出为HTML
class HtmlExportVisitor implements DocumentVisitor {
public void visit(TextElement text) {
System.out.println("导出文本段落为HTML");
}
public void visit(ImageElement image) {
System.out.println("导出图像为HTML img标签");
}
}
// 具体访问者:导出为Markdown
class MarkdownVisitor implements DocumentVisitor {
public void visit(TextElement text) {
System.out.println("导出文本段落为Markdown");
}
public void visit(ImageElement image) {
System.out.println("导出图像为Markdown图片语法");
}
}
客户端使用方式:
public class Client {
public static void main(String[] args) {
List<DocumentElement> elements = Arrays.asList(
new TextElement(), new ImageElement()
);
// 创建访问者
DocumentVisitor htmlVisitor = new HtmlExportVisitor();
DocumentVisitor mdVisitor = new MarkdownVisitor();
// 应用不同访问者
for (DocumentElement element : elements) {
element.accept(htmlVisitor);
}
for (DocumentElement element : elements) {
element.accept(mdVisitor);
}
}
}
适用场景与优缺点
适用场景:
- 对象结构相对稳定,但经常需要定义新操作
- 需要对对象结构中的元素进行多种不相关的操作
- 需要避免"污染"元素类的操作代码
优点:
- 符合开闭原则,易于添加新操作
- 将相关操作集中到访问者中,提高内聚性
- 可以跨多个类层次结构收集信息
缺点:
- 增加新元素类型困难,需要修改所有访问者
- 破坏了元素的封装性,需要暴露内部细节
- 对象结构变化成本高,可能涉及大量修改
访问者模式在编译器设计、抽象语法树处理、文档转换等场景中表现出色,是解决特定复杂问题的强大工具。

被折叠的 条评论
为什么被折叠?



